2007-11-27 19:40:25 +00:00
|
|
|
/********************************************************************
|
2007-04-29 17:35:43 +00:00
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 1999, 2000 Matthias Ettrich <ettrich@kde.org>
|
|
|
|
Copyright (C) 2003 Lubos Lunak <l.lunak@kde.org>
|
|
|
|
|
2007-11-27 19:40:25 +00:00
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*********************************************************************/
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
//#define QT_CLEAN_NAMESPACE
|
|
|
|
|
|
|
|
#include "workspace.h"
|
|
|
|
|
|
|
|
#include <kapplication.h>
|
|
|
|
#include <kstartupinfo.h>
|
|
|
|
#include <fixx11h.h>
|
|
|
|
#include <kconfig.h>
|
|
|
|
#include <kglobal.h>
|
|
|
|
#include <klocale.h>
|
|
|
|
#include <QRegExp>
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QBitmap>
|
|
|
|
#include <QClipboard>
|
|
|
|
#include <kmenubar.h>
|
2007-06-17 17:09:40 +00:00
|
|
|
#include <kprocess.h>
|
2007-04-29 17:35:43 +00:00
|
|
|
#include <kglobalaccel.h>
|
|
|
|
#include <QToolButton>
|
|
|
|
#include <kactioncollection.h>
|
|
|
|
#include <kaction.h>
|
|
|
|
#include <kconfiggroup.h>
|
2009-02-12 10:01:40 +00:00
|
|
|
#include <kcmdlineargs.h>
|
2007-04-29 17:35:43 +00:00
|
|
|
#include <QtDBus/QtDBus>
|
|
|
|
|
|
|
|
#include "client.h"
|
2010-04-25 16:43:14 +00:00
|
|
|
#include "tile.h"
|
2007-04-29 17:35:43 +00:00
|
|
|
#include "tabbox.h"
|
2009-04-06 16:09:34 +00:00
|
|
|
#include "desktopchangeosd.h"
|
2007-04-29 17:35:43 +00:00
|
|
|
#include "atoms.h"
|
|
|
|
#include "placement.h"
|
|
|
|
#include "notifications.h"
|
|
|
|
#include "group.h"
|
|
|
|
#include "rules.h"
|
|
|
|
#include "kwinadaptor.h"
|
|
|
|
#include "unmanaged.h"
|
|
|
|
#include "scene.h"
|
|
|
|
#include "deleted.h"
|
|
|
|
#include "effects.h"
|
2010-04-25 16:43:14 +00:00
|
|
|
#include "tilinglayout.h"
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2010-09-21 14:31:40 +00:00
|
|
|
#include "scripting/scripting.h"
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
#include <X11/extensions/shape.h>
|
|
|
|
#include <X11/keysym.h>
|
|
|
|
#include <X11/keysymdef.h>
|
|
|
|
#include <X11/cursorfont.h>
|
|
|
|
#include <QX11Info>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <kauthorized.h>
|
|
|
|
#include <ktoolinvocation.h>
|
|
|
|
#include <kglobalsettings.h>
|
2010-09-21 14:31:40 +00:00
|
|
|
#include <kwindowsystem.h>
|
|
|
|
#include <kwindowinfo.h>
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-11-15 03:04:44 +00:00
|
|
|
#include <kephal/screens.h>
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
extern int screen_number;
|
2011-01-13 19:05:28 +00:00
|
|
|
static const int KWIN_MAX_NUMBER_DESKTOPS = 20;
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
Workspace* Workspace::_self = 0;
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
2007-04-29 17:35:43 +00:00
|
|
|
// Rikkus: This class is too complex. It needs splitting further.
|
|
|
|
// It's a nightmare to understand, especially with so few comments :(
|
2008-12-18 13:50:57 +00:00
|
|
|
//
|
2007-04-29 17:35:43 +00:00
|
|
|
// Matthias: Feel free to ask me questions about it. Feel free to add
|
|
|
|
// comments. I dissagree that further splittings makes it easier. 2500
|
|
|
|
// lines are not too much. It's the task that is complex, not the
|
|
|
|
// code.
|
2008-12-18 13:50:57 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
Workspace::Workspace(bool restore)
|
|
|
|
: QObject(0)
|
2009-02-14 15:40:52 +00:00
|
|
|
// Desktop layout
|
2011-01-30 14:34:42 +00:00
|
|
|
, desktopCount_(0) // This is an invalid state
|
|
|
|
, desktopGridSize_(1, 2) // Default to two rows
|
|
|
|
, desktopGrid_(new int[2])
|
|
|
|
, currentDesktop_(0)
|
|
|
|
, desktopLayoutDynamicity_(false)
|
|
|
|
, tilingEnabled_(false)
|
2009-02-14 15:40:52 +00:00
|
|
|
// Unsorted
|
2011-01-30 14:34:42 +00:00
|
|
|
, active_popup(NULL)
|
|
|
|
, active_popup_client(NULL)
|
|
|
|
, temporaryRulesMessages("_KDE_NET_WM_TEMPORARY_RULES", NULL, false)
|
|
|
|
, rules_updates_disabled(false)
|
|
|
|
, active_client(0)
|
|
|
|
, last_active_client(0)
|
|
|
|
, most_recently_raised(0)
|
|
|
|
, movingClient(0)
|
|
|
|
, pending_take_activity(NULL)
|
|
|
|
, active_screen(0)
|
|
|
|
, delayfocus_client(0)
|
|
|
|
, force_restacking(false)
|
|
|
|
, x_stacking_dirty(true)
|
|
|
|
, showing_desktop(false)
|
|
|
|
, block_showing_desktop(0)
|
|
|
|
, was_user_interaction(false)
|
|
|
|
, session_saving(false)
|
|
|
|
, control_grab(false)
|
|
|
|
, tab_grab(false)
|
|
|
|
, mouse_emulation(false)
|
|
|
|
, block_focus(0)
|
|
|
|
, tab_box(0)
|
|
|
|
, desktop_change_osd(0)
|
|
|
|
, popup(0)
|
|
|
|
, advanced_popup(0)
|
|
|
|
, trans_popup(0)
|
|
|
|
, desk_popup(0)
|
|
|
|
, activity_popup(0)
|
|
|
|
, add_tabs_popup(0)
|
|
|
|
, switch_to_tab_popup(0)
|
|
|
|
, keys(0)
|
|
|
|
, client_keys(NULL)
|
|
|
|
, client_keys_dialog(NULL)
|
|
|
|
, client_keys_client(NULL)
|
|
|
|
, disable_shortcuts_keys(NULL)
|
|
|
|
, global_shortcuts_disabled(false)
|
|
|
|
, global_shortcuts_disabled_for_client(false)
|
|
|
|
, workspaceInit(true)
|
|
|
|
, startup(0)
|
|
|
|
, managing_topmenus(false)
|
|
|
|
, topmenu_selection(NULL)
|
|
|
|
, topmenu_watcher(NULL)
|
|
|
|
, topmenu_height(0)
|
|
|
|
, topmenu_space(NULL)
|
|
|
|
, set_active_client_recursion(0)
|
|
|
|
, block_stacking_updates(0)
|
|
|
|
, forced_global_mouse_grab(false)
|
|
|
|
, cm_selection(NULL)
|
|
|
|
, compositingSuspended(false)
|
|
|
|
, xrrRefreshRate(0)
|
|
|
|
, overlay(None)
|
|
|
|
, overlay_visible(true)
|
|
|
|
, overlay_shown(false)
|
|
|
|
, transSlider(NULL)
|
|
|
|
, transButton(NULL)
|
|
|
|
, forceUnredirectCheck(true)
|
|
|
|
{
|
|
|
|
(void) new KWinAdaptor(this);
|
2008-12-18 13:50:57 +00:00
|
|
|
|
2007-04-30 09:57:03 +00:00
|
|
|
QDBusConnection dbus = QDBusConnection::sessionBus();
|
2011-01-30 14:34:42 +00:00
|
|
|
dbus.registerObject("/KWin", this);
|
|
|
|
dbus.connect(QString(), "/KWin", "org.kde.KWin", "reloadConfig",
|
|
|
|
this, SLOT(slotReloadConfig()));
|
|
|
|
dbus.connect(QString(), "/KWin", "org.kde.KWin", "reinitCompositing",
|
|
|
|
this, SLOT(slotReinitCompositing()));
|
2008-12-18 13:50:57 +00:00
|
|
|
|
2009-02-14 15:40:52 +00:00
|
|
|
// Initialize desktop grid array
|
|
|
|
desktopGrid_[0] = 0;
|
|
|
|
desktopGrid_[1] = 0;
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
_self = this;
|
|
|
|
mgr = new PluginMgr;
|
2007-07-19 16:24:51 +00:00
|
|
|
QX11Info info;
|
2011-01-30 14:34:42 +00:00
|
|
|
default_colormap = DefaultColormap(display(), info.screen());
|
2007-04-29 17:35:43 +00:00
|
|
|
installed_colormap = default_colormap;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = 0; i < ELECTRIC_COUNT; ++i) {
|
2008-12-18 13:50:57 +00:00
|
|
|
electric_reserved[i] = 0;
|
|
|
|
electric_windows[i] = None;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(&temporaryRulesMessages, SIGNAL(gotMessage(const QString&)),
|
|
|
|
this, SLOT(gotTemporaryRulesMessage(const QString&)));
|
|
|
|
connect(&rulesUpdatedTimer, SIGNAL(timeout()), this, SLOT(writeWindowRules()));
|
|
|
|
connect(&unredirectTimer, SIGNAL(timeout()), this, SLOT(delayedCheckUnredirect()));
|
|
|
|
connect(&compositeResetTimer, SIGNAL(timeout()), this, SLOT(resetCompositing()));
|
|
|
|
unredirectTimer.setSingleShot(true);
|
|
|
|
compositeResetTimer.setSingleShot(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
updateXTime(); // Needed for proper initialization of user_time in Client ctor
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
delayFocusTimer = 0;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (restore)
|
2008-12-18 13:50:57 +00:00
|
|
|
loadSessionInfo();
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
loadWindowRules();
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Call this before XSelectInput() on the root window
|
2007-04-29 17:35:43 +00:00
|
|
|
startup = new KStartupInfo(
|
2011-01-30 14:34:42 +00:00
|
|
|
KStartupInfo::DisableKWinModule | KStartupInfo::AnnounceSilenceChanges, this);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Select windowmanager privileges
|
2011-01-30 14:34:42 +00:00
|
|
|
XSelectInput(display(), rootWindow(),
|
|
|
|
KeyPressMask |
|
|
|
|
PropertyChangeMask |
|
|
|
|
ColormapChangeMask |
|
|
|
|
SubstructureRedirectMask |
|
|
|
|
SubstructureNotifyMask |
|
|
|
|
FocusChangeMask | // For NotifyDetailNone
|
|
|
|
ExposureMask
|
|
|
|
);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
Extensions::init();
|
|
|
|
setupCompositing();
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Compatibility
|
2007-04-29 17:35:43 +00:00
|
|
|
long data = 1;
|
|
|
|
|
|
|
|
XChangeProperty(
|
2008-12-18 13:50:57 +00:00
|
|
|
display(),
|
|
|
|
rootWindow(),
|
|
|
|
atoms->kwin_running,
|
|
|
|
atoms->kwin_running,
|
|
|
|
32,
|
|
|
|
PropModeAppend,
|
2011-01-30 14:34:42 +00:00
|
|
|
(unsigned char*)(&data),
|
2008-12-18 13:50:57 +00:00
|
|
|
1
|
2011-01-30 14:34:42 +00:00
|
|
|
);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
client_keys = new KActionCollection(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
initShortcuts();
|
2011-01-30 14:34:42 +00:00
|
|
|
tab_box = new TabBox::TabBox(this);
|
|
|
|
desktop_change_osd = new DesktopChangeOSD(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
init();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(Kephal::Screens::self(), SIGNAL(screenAdded(Kephal::Screen*)), SLOT(desktopResized()));
|
|
|
|
connect(Kephal::Screens::self(), SIGNAL(screenRemoved(int)), SLOT(desktopResized()));
|
|
|
|
connect(Kephal::Screens::self(), SIGNAL(screenResized(Kephal::Screen*, QSize, QSize)), SLOT(desktopResized()));
|
|
|
|
connect(Kephal::Screens::self(), SIGNAL(screenMoved(Kephal::Screen*, QPoint, QPoint)), SLOT(desktopResized()));
|
2010-05-11 20:30:20 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(&activityController_, SIGNAL(currentActivityChanged(QString)), SLOT(updateCurrentActivity(QString)));
|
|
|
|
connect(&activityController_, SIGNAL(activityRemoved(QString)), SLOT(activityRemoved(QString)));
|
|
|
|
connect(&activityController_, SIGNAL(activityAdded(QString)), SLOT(activityAdded(QString)));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::init()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
reserveElectricBorderActions(true);
|
|
|
|
if (options->electricBorders() == Options::ElectricAlways)
|
|
|
|
reserveElectricBorderSwitching(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
updateElectricBorders();
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Not used yet
|
|
|
|
//topDock = 0L;
|
|
|
|
//maximizedWindowCounter = 0;
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
supportWindow = new QWidget(NULL, Qt::X11BypassWindowManagerHint);
|
|
|
|
XLowerWindow(display(), supportWindow->winId()); // See usage in layers.cpp
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
XSetWindowAttributes attr;
|
|
|
|
attr.override_redirect = 1;
|
2011-01-30 14:34:42 +00:00
|
|
|
null_focus_window = XCreateWindow(display(), rootWindow(), -1, -1, 1, 1, 0, CopyFromParent,
|
|
|
|
InputOnly, CopyFromParent, CWOverrideRedirect, &attr);
|
|
|
|
XMapWindow(display(), null_focus_window);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
unsigned long protocols[5] = {
|
2007-04-29 17:35:43 +00:00
|
|
|
NET::Supported |
|
|
|
|
NET::SupportingWMCheck |
|
|
|
|
NET::ClientList |
|
|
|
|
NET::ClientListStacking |
|
|
|
|
NET::DesktopGeometry |
|
|
|
|
NET::NumberOfDesktops |
|
|
|
|
NET::CurrentDesktop |
|
|
|
|
NET::ActiveWindow |
|
|
|
|
NET::WorkArea |
|
|
|
|
NET::CloseWindow |
|
|
|
|
NET::DesktopNames |
|
|
|
|
NET::WMName |
|
|
|
|
NET::WMVisibleName |
|
|
|
|
NET::WMDesktop |
|
|
|
|
NET::WMWindowType |
|
|
|
|
NET::WMState |
|
|
|
|
NET::WMStrut |
|
|
|
|
NET::WMIconGeometry |
|
|
|
|
NET::WMIcon |
|
|
|
|
NET::WMPid |
|
|
|
|
NET::WMMoveResize |
|
|
|
|
NET::WMFrameExtents |
|
|
|
|
NET::WMPing
|
|
|
|
,
|
|
|
|
NET::NormalMask |
|
|
|
|
NET::DesktopMask |
|
|
|
|
NET::DockMask |
|
|
|
|
NET::ToolbarMask |
|
|
|
|
NET::MenuMask |
|
|
|
|
NET::DialogMask |
|
|
|
|
NET::OverrideMask |
|
|
|
|
NET::TopMenuMask |
|
|
|
|
NET::UtilityMask |
|
|
|
|
NET::SplashMask |
|
2008-12-18 13:50:57 +00:00
|
|
|
// No compositing window types here unless we support them also as managed window types
|
2007-04-29 17:35:43 +00:00
|
|
|
0
|
|
|
|
,
|
|
|
|
NET::Modal |
|
2008-12-18 13:50:57 +00:00
|
|
|
//NET::Sticky | // Large desktops not supported (and probably never will be)
|
2007-04-29 17:35:43 +00:00
|
|
|
NET::MaxVert |
|
|
|
|
NET::MaxHoriz |
|
|
|
|
NET::Shaded |
|
|
|
|
NET::SkipTaskbar |
|
|
|
|
NET::KeepAbove |
|
2008-12-18 13:50:57 +00:00
|
|
|
//NET::StaysOnTop | // The same like KeepAbove
|
2007-04-29 17:35:43 +00:00
|
|
|
NET::SkipPager |
|
|
|
|
NET::Hidden |
|
|
|
|
NET::FullScreen |
|
|
|
|
NET::KeepBelow |
|
|
|
|
NET::DemandsAttention |
|
|
|
|
0
|
|
|
|
,
|
|
|
|
NET::WM2UserTime |
|
|
|
|
NET::WM2StartupId |
|
|
|
|
NET::WM2AllowedActions |
|
|
|
|
NET::WM2RestackWindow |
|
|
|
|
NET::WM2MoveResizeWindow |
|
|
|
|
NET::WM2ExtendedStrut |
|
|
|
|
NET::WM2KDETemporaryRules |
|
|
|
|
NET::WM2ShowingDesktop |
|
2007-04-30 12:00:32 +00:00
|
|
|
NET::WM2DesktopLayout |
|
2007-05-07 12:37:12 +00:00
|
|
|
NET::WM2FullPlacement |
|
2009-01-02 00:40:05 +00:00
|
|
|
NET::WM2FullscreenMonitors |
|
2007-04-29 17:35:43 +00:00
|
|
|
0
|
|
|
|
,
|
|
|
|
NET::ActionMove |
|
|
|
|
NET::ActionResize |
|
|
|
|
NET::ActionMinimize |
|
|
|
|
NET::ActionShade |
|
2008-12-18 13:50:57 +00:00
|
|
|
//NET::ActionStick | // Sticky state is not supported
|
2007-04-29 17:35:43 +00:00
|
|
|
NET::ActionMaxVert |
|
|
|
|
NET::ActionMaxHoriz |
|
|
|
|
NET::ActionFullScreen |
|
|
|
|
NET::ActionChangeDesktop |
|
|
|
|
NET::ActionClose |
|
|
|
|
0
|
|
|
|
,
|
2011-01-30 14:34:42 +00:00
|
|
|
};
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (mgr->factory()->supports(AbilityExtendIntoClientArea))
|
2009-12-16 18:03:15 +00:00
|
|
|
protocols[ NETRootInfo::PROTOCOLS2 ] |= NET::WM2FrameOverlap;
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
QX11Info info;
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo = new RootInfo(this, display(), supportWindow->winId(), "KWin", protocols, 5, info.screen());
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
loadDesktopSettings();
|
2007-05-30 18:17:18 +00:00
|
|
|
updateDesktopLayout();
|
2009-04-06 16:09:34 +00:00
|
|
|
desktop_change_osd->numberDesktopsChanged();
|
2008-12-18 13:50:57 +00:00
|
|
|
// Extra NETRootInfo instance in Client mode is needed to get the values of the properties
|
2011-01-30 14:34:42 +00:00
|
|
|
NETRootInfo client_info(display(), NET::ActiveWindow | NET::CurrentDesktop);
|
2007-04-29 17:35:43 +00:00
|
|
|
int initial_desktop;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!kapp->isSessionRestored())
|
2007-04-29 17:35:43 +00:00
|
|
|
initial_desktop = client_info.currentDesktop();
|
2011-01-30 14:34:42 +00:00
|
|
|
else {
|
|
|
|
KConfigGroup group(kapp->sessionConfig(), "Session");
|
|
|
|
initial_desktop = group.readEntry("desktop", 1);
|
|
|
|
}
|
|
|
|
if (!setCurrentDesktop(initial_desktop))
|
|
|
|
setCurrentDesktop(1);
|
2010-11-01 19:53:22 +00:00
|
|
|
allActivities_ = activityController_.listActivities();
|
2011-01-30 14:34:42 +00:00
|
|
|
updateCurrentActivity(activityController_.currentActivity());
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Now we know how many desktops we'll have, thus we initialize the positioning object
|
2011-01-30 14:34:42 +00:00
|
|
|
initPositioning = new Placement(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
reconfigureTimer.setSingleShot(true);
|
|
|
|
updateToolWindowsTimer.setSingleShot(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(&reconfigureTimer, SIGNAL(timeout()), this, SLOT(slotReconfigure()));
|
|
|
|
connect(&updateToolWindowsTimer, SIGNAL(timeout()), this, SLOT(slotUpdateToolWindows()));
|
|
|
|
connect(&mousePollingTimer, SIGNAL(timeout()), SLOT(performMousePoll()));
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(KGlobalSettings::self(), SIGNAL(appearanceChanged()), this, SLOT(reconfigure()));
|
|
|
|
connect(KGlobalSettings::self(), SIGNAL(settingsChanged(int)), this, SLOT(slotSettingsChanged(int)));
|
|
|
|
connect(KGlobalSettings::self(), SIGNAL(blockShortcuts(int)), this, SLOT(slotBlockShortcuts(int)));
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
active_client = NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setActiveWindow(None);
|
2007-04-29 17:35:43 +00:00
|
|
|
focusToNull();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!kapp->isSessionRestored())
|
2008-12-18 13:50:57 +00:00
|
|
|
++block_focus; // Because it will be set below
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
char nm[100];
|
2011-01-30 14:34:42 +00:00
|
|
|
sprintf(nm, "_KDE_TOPMENU_OWNER_S%d", DefaultScreen(display()));
|
|
|
|
Atom topmenu_atom = XInternAtom(display(), nm, False);
|
|
|
|
topmenu_selection = new KSelectionOwner(topmenu_atom);
|
|
|
|
topmenu_watcher = new KSelectionWatcher(topmenu_atom);
|
2008-12-18 13:50:57 +00:00
|
|
|
//TODO: grabXServer(); // Where exactly put this? topmenu selection claiming down belong must be before
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
// Begin updates blocker block
|
|
|
|
StackingUpdatesBlocker blocker(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (options->topMenuEnabled() && topmenu_selection->claim(false))
|
2008-12-18 13:50:57 +00:00
|
|
|
setupTopMenuHandling(); // This can call updateStackingOrder()
|
2007-04-29 17:35:43 +00:00
|
|
|
else
|
|
|
|
lostTopMenuSelection();
|
|
|
|
|
|
|
|
unsigned int i, nwins;
|
2008-12-18 13:50:57 +00:00
|
|
|
Window root_return, parent_return;
|
|
|
|
Window* wins;
|
2011-01-30 14:34:42 +00:00
|
|
|
XQueryTree(display(), rootWindow(), &root_return, &parent_return, &wins, &nwins);
|
|
|
|
bool fixoffset = KCmdLineArgs::parsedArgs()->getOption("crashes").toInt() > 0;
|
|
|
|
for (i = 0; i < nwins; i++) {
|
2007-04-29 17:35:43 +00:00
|
|
|
XWindowAttributes attr;
|
2011-01-30 14:34:42 +00:00
|
|
|
XGetWindowAttributes(display(), wins[i], &attr);
|
|
|
|
if (attr.override_redirect) {
|
|
|
|
createUnmanaged(wins[i]);
|
2007-04-29 17:35:43 +00:00
|
|
|
continue;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (topmenu_space && topmenu_space->winId() == wins[i])
|
2007-04-29 17:35:43 +00:00
|
|
|
continue;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (attr.map_state != IsUnmapped) {
|
|
|
|
if (fixoffset)
|
|
|
|
fixPositionAfterCrash(wins[ i ], attr);
|
|
|
|
createClient(wins[i], true);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (wins)
|
|
|
|
XFree((void*)(wins));
|
2008-12-18 13:50:57 +00:00
|
|
|
|
|
|
|
// Propagate clients, will really happen at the end of the updates blocker block
|
2011-01-30 14:34:42 +00:00
|
|
|
updateStackingOrder(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
updateClientArea();
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// NETWM spec says we have to set it to (0,0) if we don't support it
|
2009-02-14 09:46:12 +00:00
|
|
|
NETPoint* viewports = new NETPoint[numberOfDesktops()];
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setDesktopViewport(numberOfDesktops(), *viewports);
|
2007-04-29 17:35:43 +00:00
|
|
|
delete[] viewports;
|
2008-11-15 03:04:44 +00:00
|
|
|
QRect geom = Kephal::ScreenUtils::desktopGeometry();
|
2007-04-29 17:35:43 +00:00
|
|
|
NETSize desktop_geometry;
|
|
|
|
desktop_geometry.width = geom.width();
|
|
|
|
desktop_geometry.height = geom.height();
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setDesktopGeometry(-1, desktop_geometry);
|
|
|
|
setShowingDesktop(false);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
} // End updates blocker block
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
Client* new_active_client = NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!kapp->isSessionRestored()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
--block_focus;
|
2011-01-30 14:34:42 +00:00
|
|
|
new_active_client = findClient(WindowMatchPredicate(client_info.activeWindow()));
|
|
|
|
}
|
|
|
|
if (new_active_client == NULL
|
|
|
|
&& activeClient() == NULL && should_get_focus.count() == 0) {
|
|
|
|
// No client activated in manage()
|
|
|
|
if (new_active_client == NULL)
|
|
|
|
new_active_client = topClientOnDesktop(currentDesktop(), -1);
|
|
|
|
if (new_active_client == NULL && !desktops.isEmpty())
|
|
|
|
new_active_client = findDesktop(true, currentDesktop());
|
|
|
|
}
|
|
|
|
if (new_active_client != NULL)
|
|
|
|
activateClient(new_active_client);
|
2009-09-08 20:01:08 +00:00
|
|
|
|
|
|
|
// outline windows for electric border maximize window mode
|
2011-01-30 14:34:42 +00:00
|
|
|
outline_left = XCreateWindow(QX11Info::display(), QX11Info::appRootWindow(), 0, 0, 1, 1, 0,
|
|
|
|
CopyFromParent, CopyFromParent, CopyFromParent, CWOverrideRedirect, &attr);
|
|
|
|
outline_right = XCreateWindow(QX11Info::display(), QX11Info::appRootWindow(), 0, 0, 1, 1, 0,
|
|
|
|
CopyFromParent, CopyFromParent, CopyFromParent, CWOverrideRedirect, &attr);
|
|
|
|
outline_top = XCreateWindow(QX11Info::display(), QX11Info::appRootWindow(), 0, 0, 1, 1, 0,
|
|
|
|
CopyFromParent, CopyFromParent, CopyFromParent, CWOverrideRedirect, &attr);
|
|
|
|
outline_bottom = XCreateWindow(QX11Info::display(), QX11Info::appRootWindow(), 0, 0, 1, 1, 0,
|
|
|
|
CopyFromParent, CopyFromParent, CopyFromParent, CWOverrideRedirect, &attr);
|
2009-09-08 20:01:08 +00:00
|
|
|
|
2010-04-25 16:43:14 +00:00
|
|
|
// Enable/disable tiling
|
2011-01-30 14:34:42 +00:00
|
|
|
setTilingEnabled(options->tilingOn);
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// SELI TODO: This won't work with unreasonable focus policies,
|
2007-04-29 17:35:43 +00:00
|
|
|
// and maybe in rare cases also if the selected client doesn't
|
|
|
|
// want focus
|
|
|
|
workspaceInit = false;
|
2008-12-18 13:50:57 +00:00
|
|
|
|
|
|
|
// TODO: ungrabXServer()
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
Workspace::~Workspace()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
finishCompositing();
|
2011-01-30 14:34:42 +00:00
|
|
|
blockStackingUpdates(true);
|
2008-12-18 13:50:57 +00:00
|
|
|
|
|
|
|
// TODO: grabXServer();
|
|
|
|
|
|
|
|
// Use stacking_order, so that kwin --replace keeps stacking order
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = stacking_order.constBegin();
|
|
|
|
it != stacking_order.constEnd();
|
|
|
|
++it) {
|
2008-12-18 13:50:57 +00:00
|
|
|
// Only release the window
|
2011-01-30 14:34:42 +00:00
|
|
|
(*it)->releaseWindow(true);
|
2007-04-30 09:46:31 +00:00
|
|
|
// No removeClient() is called, it does more than just removing.
|
|
|
|
// However, remove from some lists to e.g. prevent performTransiencyCheck()
|
|
|
|
// from crashing.
|
2011-01-30 14:34:42 +00:00
|
|
|
clients.removeAll(*it);
|
|
|
|
desktops.removeAll(*it);
|
|
|
|
}
|
|
|
|
for (UnmanagedList::ConstIterator it = unmanaged.constBegin();
|
|
|
|
it != unmanaged.constEnd();
|
|
|
|
++it)
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->release();
|
|
|
|
delete tab_box;
|
2009-04-06 16:09:34 +00:00
|
|
|
delete desktop_change_osd;
|
2008-08-07 11:51:20 +00:00
|
|
|
discardPopup();
|
2011-01-30 14:34:42 +00:00
|
|
|
XDeleteProperty(display(), rootWindow(), atoms->kwin_running);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
writeWindowRules();
|
|
|
|
KGlobal::config()->sync();
|
|
|
|
|
2009-09-08 20:01:08 +00:00
|
|
|
// destroy outline windows for electric border maximize window mode
|
2011-01-30 14:34:42 +00:00
|
|
|
XDestroyWindow(QX11Info::display(), outline_left);
|
|
|
|
XDestroyWindow(QX11Info::display(), outline_right);
|
|
|
|
XDestroyWindow(QX11Info::display(), outline_top);
|
|
|
|
XDestroyWindow(QX11Info::display(), outline_bottom);
|
2009-09-08 20:01:08 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
delete rootInfo;
|
|
|
|
delete supportWindow;
|
|
|
|
delete mgr;
|
|
|
|
delete startup;
|
|
|
|
delete initPositioning;
|
|
|
|
delete topmenu_watcher;
|
|
|
|
delete topmenu_selection;
|
|
|
|
delete topmenu_space;
|
2007-11-01 19:24:35 +00:00
|
|
|
delete client_keys_dialog;
|
2011-01-30 14:34:42 +00:00
|
|
|
while (!rules.isEmpty()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
delete rules.front();
|
|
|
|
rules.pop_front();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
foreach (SessionInfo * s, session)
|
|
|
|
delete s;
|
|
|
|
XDestroyWindow(display(), null_focus_window);
|
2008-12-18 13:50:57 +00:00
|
|
|
|
|
|
|
// TODO: ungrabXServer();
|
|
|
|
|
2009-02-14 15:40:52 +00:00
|
|
|
delete[] desktopGrid_;
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
_self = 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
Client* Workspace::createClient(Window w, bool is_mapped)
|
|
|
|
{
|
|
|
|
StackingUpdatesBlocker blocker(this);
|
|
|
|
Client* c = new Client(this);
|
|
|
|
if (!c->manage(w, is_mapped)) {
|
|
|
|
Client::deleteClient(c, Allowed);
|
2007-04-29 17:35:43 +00:00
|
|
|
return NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
addClient(c, Allowed);
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
tilingLayouts.resize(numberOfDesktops() + 1);
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
createTile(c);
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (scene)
|
|
|
|
scene->windowAdded(c);
|
|
|
|
if (effects)
|
|
|
|
static_cast<EffectsHandlerImpl*>(effects)->windowAdded(c->effectWindow());
|
2007-04-29 17:35:43 +00:00
|
|
|
return c;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
Unmanaged* Workspace::createUnmanaged(Window w)
|
|
|
|
{
|
|
|
|
if (w == overlay)
|
2007-04-29 17:35:43 +00:00
|
|
|
return NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
Unmanaged* c = new Unmanaged(this);
|
|
|
|
if (!c->track(w)) {
|
|
|
|
Unmanaged::deleteUnmanaged(c, Allowed);
|
2007-04-29 17:35:43 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
addUnmanaged(c, Allowed);
|
|
|
|
if (scene)
|
|
|
|
scene->windowAdded(c);
|
|
|
|
if (effects)
|
|
|
|
static_cast<EffectsHandlerImpl*>(effects)->windowAdded(c->effectWindow());
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Workspace::addClient(Client* c, allowed_t)
|
|
|
|
{
|
|
|
|
Group* grp = findGroup(c->window());
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2010-09-21 14:31:40 +00:00
|
|
|
KWindowInfo info = KWindowSystem::windowInfo(c->window(), -1U, NET::WM2WindowClass);
|
2011-01-30 14:34:42 +00:00
|
|
|
|
2010-09-21 14:31:40 +00:00
|
|
|
/*
|
2011-01-30 14:34:42 +00:00
|
|
|
if (info.windowClassName() == QString("krunner")) {
|
|
|
|
SWrapper::Workspace* ws_object = KWin::Scripting::workspace();
|
|
|
|
if (ws_object != 0) {
|
|
|
|
ws_object->sl_killWindowCalled(c);
|
|
|
|
}
|
2010-09-21 14:31:40 +00:00
|
|
|
}*/
|
2011-01-30 14:34:42 +00:00
|
|
|
|
2010-09-21 14:31:40 +00:00
|
|
|
emit clientAdded(c);
|
2011-01-30 14:34:42 +00:00
|
|
|
|
|
|
|
if (grp != NULL)
|
|
|
|
grp->gotLeader(c);
|
|
|
|
|
|
|
|
if (c->isDesktop()) {
|
|
|
|
desktops.append(c);
|
|
|
|
if (active_client == NULL && should_get_focus.isEmpty() && c->isOnCurrentDesktop())
|
|
|
|
requestFocus(c); // TODO: Make sure desktop is active after startup if there's no other window active
|
|
|
|
} else {
|
|
|
|
updateFocusChains(c, FocusChainUpdate); // Add to focus chain if not already there
|
|
|
|
clients.append(c);
|
|
|
|
}
|
|
|
|
if (!unconstrained_stacking_order.contains(c))
|
|
|
|
unconstrained_stacking_order.append(c); // Raise if it hasn't got any stacking position yet
|
|
|
|
if (!stacking_order.contains(c)) // It'll be updated later, and updateToolWindows() requires
|
|
|
|
stacking_order.append(c); // c to be in stacking_order
|
|
|
|
if (c->isTopMenu())
|
|
|
|
addTopMenu(c);
|
2008-08-24 10:35:45 +00:00
|
|
|
x_stacking_dirty = true;
|
2008-12-18 13:50:57 +00:00
|
|
|
updateClientArea(); // This cannot be in manage(), because the client got added only now
|
2011-01-30 14:34:42 +00:00
|
|
|
updateClientLayer(c);
|
|
|
|
if (c->isDesktop()) {
|
|
|
|
raiseClient(c);
|
2008-12-18 13:50:57 +00:00
|
|
|
// If there's no active client, make this desktop the active one
|
2011-01-30 14:34:42 +00:00
|
|
|
if (activeClient() == NULL && should_get_focus.count() == 0)
|
|
|
|
activateClient(findDesktop(true, currentDesktop()));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
c->checkActiveModal();
|
2011-01-30 14:34:42 +00:00
|
|
|
checkTransients(c->window()); // SELI TODO: Does this really belong here?
|
|
|
|
updateStackingOrder(true); // Propagate new client
|
|
|
|
if (c->isUtility() || c->isMenu() || c->isToolbar())
|
|
|
|
updateToolWindows(true);
|
2007-04-30 09:47:59 +00:00
|
|
|
checkNonExistentClients();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (tab_grab)
|
|
|
|
tab_box->reset(true);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::addUnmanaged(Unmanaged* c, allowed_t)
|
|
|
|
{
|
|
|
|
unmanaged.append(c);
|
2008-08-24 10:35:45 +00:00
|
|
|
x_stacking_dirty = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Destroys the client \a c
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::removeClient(Client* c, allowed_t)
|
|
|
|
{
|
2010-09-21 14:31:40 +00:00
|
|
|
emit clientRemoved(c);
|
2011-01-30 14:34:42 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
if (c == active_popup_client)
|
|
|
|
closeActivePopup();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (client_keys_client == c)
|
|
|
|
setupWindowShortcutDone(false);
|
|
|
|
if (!c->shortcut().isEmpty()) {
|
|
|
|
c->setShortcut(QString()); // Remove from client_keys
|
|
|
|
clientShortcutUpdated(c); // Needed, since this is otherwise delayed by setShortcut() and wouldn't run
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c->isDialog())
|
|
|
|
Notify::raise(Notify::TransDelete);
|
|
|
|
if (c->isNormalWindow())
|
|
|
|
Notify::raise(Notify::Delete);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (tab_grab && tab_box->currentClient() == c)
|
|
|
|
tab_box->nextPrev(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
Q_ASSERT(clients.contains(c) || desktops.contains(c));
|
|
|
|
if (tilingEnabled() && tilingLayouts.value(c->desktop())) {
|
|
|
|
removeTile(c);
|
|
|
|
}
|
2010-04-25 16:43:14 +00:00
|
|
|
// TODO: if marked client is removed, notify the marked list
|
2011-01-30 14:34:42 +00:00
|
|
|
clients.removeAll(c);
|
|
|
|
desktops.removeAll(c);
|
|
|
|
unconstrained_stacking_order.removeAll(c);
|
|
|
|
stacking_order.removeAll(c);
|
2008-08-24 10:35:45 +00:00
|
|
|
x_stacking_dirty = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = 1; i <= numberOfDesktops(); ++i)
|
|
|
|
focus_chain[i].removeAll(c);
|
|
|
|
global_focus_chain.removeAll(c);
|
|
|
|
attention_chain.removeAll(c);
|
|
|
|
showing_desktop_clients.removeAll(c);
|
|
|
|
if (c->isTopMenu())
|
|
|
|
removeTopMenu(c);
|
|
|
|
Group* group = findGroup(c->window());
|
|
|
|
if (group != NULL)
|
2007-04-29 17:35:43 +00:00
|
|
|
group->lostLeader();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c == most_recently_raised)
|
2007-04-29 17:35:43 +00:00
|
|
|
most_recently_raised = 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
should_get_focus.removeAll(c);
|
|
|
|
Q_ASSERT(c != active_client);
|
|
|
|
if (c == last_active_client)
|
2007-04-29 17:35:43 +00:00
|
|
|
last_active_client = 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c == pending_take_activity)
|
2007-04-29 17:35:43 +00:00
|
|
|
pending_take_activity = NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c == delayfocus_client)
|
2007-04-29 17:35:43 +00:00
|
|
|
cancelDelayFocus();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
updateStackingOrder(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (tab_grab)
|
|
|
|
tab_box->reset(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
updateClientArea();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::removeUnmanaged(Unmanaged* c, allowed_t)
|
|
|
|
{
|
|
|
|
assert(unmanaged.contains(c));
|
|
|
|
unmanaged.removeAll(c);
|
2008-08-24 10:35:45 +00:00
|
|
|
x_stacking_dirty = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::addDeleted(Deleted* c, allowed_t)
|
|
|
|
{
|
|
|
|
assert(!deleted.contains(c));
|
|
|
|
deleted.append(c);
|
2008-08-24 10:35:45 +00:00
|
|
|
x_stacking_dirty = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::removeDeleted(Deleted* c, allowed_t)
|
|
|
|
{
|
|
|
|
assert(deleted.contains(c));
|
|
|
|
if (scene)
|
|
|
|
scene->windowDeleted(c);
|
|
|
|
if (effects)
|
|
|
|
static_cast<EffectsHandlerImpl*>(effects)->windowDeleted(c->effectWindow());
|
|
|
|
deleted.removeAll(c);
|
2008-08-24 10:35:45 +00:00
|
|
|
x_stacking_dirty = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::updateFocusChains(Client* c, FocusChainChange change)
|
|
|
|
{
|
|
|
|
if (!c->wantsTabFocus()) { // Doesn't want tab focus, remove
|
|
|
|
for (int i = 1; i <= numberOfDesktops(); ++i)
|
|
|
|
focus_chain[i].removeAll(c);
|
|
|
|
global_focus_chain.removeAll(c);
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (c->desktop() == NET::OnAllDesktops) {
|
|
|
|
// Now on all desktops, add it to focus_chains it is not already in
|
|
|
|
for (int i = 1; i <= numberOfDesktops(); i++) {
|
|
|
|
// Making first/last works only on current desktop, don't affect all desktops
|
|
|
|
if (i == currentDesktop()
|
|
|
|
&& (change == FocusChainMakeFirst || change == FocusChainMakeLast)) {
|
|
|
|
focus_chain[i].removeAll(c);
|
|
|
|
if (change == FocusChainMakeFirst)
|
|
|
|
focus_chain[i].append(c);
|
2007-04-29 17:35:43 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
focus_chain[i].prepend(c);
|
|
|
|
} else if (!focus_chain[i].contains(c)) {
|
|
|
|
// Add it after the active one
|
|
|
|
if (active_client != NULL && active_client != c &&
|
|
|
|
!focus_chain[i].isEmpty() && focus_chain[i].last() == active_client)
|
|
|
|
focus_chain[i].insert(focus_chain[i].size() - 1, c);
|
2007-04-29 22:33:55 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
focus_chain[i].append(c); // Otherwise add as the first one
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
} else { // Now only on desktop, remove it anywhere else
|
|
|
|
for (int i = 1; i <= numberOfDesktops(); i++) {
|
|
|
|
if (i == c->desktop()) {
|
|
|
|
if (change == FocusChainMakeFirst) {
|
|
|
|
focus_chain[i].removeAll(c);
|
|
|
|
focus_chain[i].append(c);
|
|
|
|
} else if (change == FocusChainMakeLast) {
|
|
|
|
focus_chain[i].removeAll(c);
|
|
|
|
focus_chain[i].prepend(c);
|
|
|
|
} else if (!focus_chain[i].contains(c)) {
|
|
|
|
// Add it after the active one
|
|
|
|
if (active_client != NULL && active_client != c &&
|
|
|
|
!focus_chain[i].isEmpty() && focus_chain[i].last() == active_client)
|
|
|
|
focus_chain[i].insert(focus_chain[i].size() - 1, c);
|
2007-04-29 22:33:55 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
focus_chain[i].append(c); // Otherwise add as the first one
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
} else
|
|
|
|
focus_chain[i].removeAll(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (change == FocusChainMakeFirst) {
|
|
|
|
global_focus_chain.removeAll(c);
|
|
|
|
global_focus_chain.append(c);
|
|
|
|
} else if (change == FocusChainMakeLast) {
|
|
|
|
global_focus_chain.removeAll(c);
|
|
|
|
global_focus_chain.prepend(c);
|
|
|
|
} else if (!global_focus_chain.contains(c)) {
|
|
|
|
// Add it after the active one
|
|
|
|
if (active_client != NULL && active_client != c &&
|
|
|
|
!global_focus_chain.isEmpty() && global_focus_chain.last() == active_client)
|
|
|
|
global_focus_chain.insert(global_focus_chain.size() - 1, c);
|
2007-04-29 22:33:55 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
global_focus_chain.append(c); // Otherwise add as the first one
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::updateCurrentTopMenu()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (!managingTopMenus())
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
|
|
|
// toplevel menubar handling
|
|
|
|
Client* menubar = 0;
|
|
|
|
bool block_desktop_menubar = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (active_client) {
|
2008-12-18 13:50:57 +00:00
|
|
|
// Show the new menu bar first...
|
2007-04-29 17:35:43 +00:00
|
|
|
Client* menu_client = active_client;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (;;) {
|
|
|
|
if (menu_client->isFullScreen())
|
2007-04-29 17:35:43 +00:00
|
|
|
block_desktop_menubar = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = menu_client->transients().constBegin();
|
|
|
|
it != menu_client->transients().constEnd();
|
|
|
|
++it)
|
|
|
|
if ((*it)->isTopMenu()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
menubar = *it;
|
|
|
|
break;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (menubar != NULL || !menu_client->isTransient())
|
2007-04-29 17:35:43 +00:00
|
|
|
break;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (menu_client->isModal() || menu_client->transientFor() == NULL)
|
2008-12-18 13:50:57 +00:00
|
|
|
break; // Don't use mainwindow's menu if this is modal or group transient
|
2007-04-29 17:35:43 +00:00
|
|
|
menu_client = menu_client->transientFor();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (!menubar) {
|
|
|
|
// Try to find any topmenu from the application (#72113)
|
|
|
|
for (ClientList::ConstIterator it = active_client->group()->members().constBegin();
|
|
|
|
it != active_client->group()->members().constEnd();
|
|
|
|
++it)
|
|
|
|
if ((*it)->isTopMenu()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
menubar = *it;
|
|
|
|
break;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (!menubar && !block_desktop_menubar && options->desktopTopMenu()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
// Find the menubar of the desktop
|
2011-01-30 14:34:42 +00:00
|
|
|
Client* desktop = findDesktop(true, currentDesktop());
|
|
|
|
if (desktop != NULL) {
|
|
|
|
for (ClientList::ConstIterator it = desktop->transients().constBegin();
|
|
|
|
it != desktop->transients().constEnd();
|
|
|
|
++it)
|
|
|
|
if ((*it)->isTopMenu()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
menubar = *it;
|
|
|
|
break;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
// TODO: To be cleaned app with window grouping
|
2007-04-29 17:35:43 +00:00
|
|
|
// Without qt-copy patch #0009, the topmenu and desktop are not in the same group,
|
|
|
|
// thus the topmenu is not transient for it :-/.
|
2011-01-30 14:34:42 +00:00
|
|
|
if (menubar == NULL) {
|
|
|
|
for (ClientList::ConstIterator it = topmenus.constBegin();
|
|
|
|
it != topmenus.constEnd();
|
|
|
|
++it)
|
2008-12-18 13:50:57 +00:00
|
|
|
// kdesktop's topmenu has WM_TRANSIENT_FOR set pointing to the root window
|
|
|
|
// to recognize it here. Also, with the xroot hack in kdesktop, there's
|
|
|
|
// no NET::Desktop window to be transient for.
|
2011-01-30 14:34:42 +00:00
|
|
|
if ((*it)->wasOriginallyGroupTransient()) {
|
2008-12-18 13:50:57 +00:00
|
|
|
menubar = *it;
|
|
|
|
break;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
//kDebug( 1212 ) << "CURRENT TOPMENU:" << menubar << ":" << active_client;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (menubar) {
|
|
|
|
if (active_client && !menubar->isOnDesktop(active_client->desktop()))
|
|
|
|
menubar->setDesktop(active_client->desktop());
|
|
|
|
menubar->hideClient(false);
|
2007-04-29 17:35:43 +00:00
|
|
|
topmenu_space->hide();
|
2008-12-18 13:50:57 +00:00
|
|
|
// Make it appear like it's been raised manually - it's in the Dock layer anyway,
|
2007-04-29 17:35:43 +00:00
|
|
|
// and not raising it could mess up stacking order of topmenus within one application,
|
|
|
|
// and thus break raising of mainclients in raiseClient()
|
2011-01-30 14:34:42 +00:00
|
|
|
unconstrained_stacking_order.removeAll(menubar);
|
|
|
|
unconstrained_stacking_order.append(menubar);
|
|
|
|
} else if (!block_desktop_menubar) {
|
|
|
|
// No topmenu active - show the space window, so that there's not empty space
|
2007-04-29 17:35:43 +00:00
|
|
|
topmenu_space->show();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// ... Then hide the other ones. Avoids flickers.
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = clients.constBegin(); it != clients.constEnd(); ++it)
|
|
|
|
if ((*it)->isTopMenu() && (*it) != menubar)
|
|
|
|
(*it)->hideClient(true);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::updateToolWindows(bool also_hide)
|
|
|
|
{
|
2008-12-18 13:50:57 +00:00
|
|
|
// TODO: What if Client's transiency/group changes? should this be called too? (I'm paranoid, am I not?)
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!options->hideUtilityWindowsForInactive) {
|
|
|
|
for (ClientList::ConstIterator it = clients.constBegin();
|
|
|
|
it != clients.constEnd();
|
|
|
|
++it)
|
|
|
|
(*it)->hideClient(false);
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
const Group* group = NULL;
|
|
|
|
const Client* client = active_client;
|
2008-12-18 13:50:57 +00:00
|
|
|
// Go up in transiency hiearchy, if the top is found, only tool transients for the top mainwindow
|
|
|
|
// will be shown; if a group transient is group, all tools in the group will be shown
|
2011-01-30 14:34:42 +00:00
|
|
|
while (client != NULL) {
|
|
|
|
if (!client->isTransient())
|
2007-04-29 17:35:43 +00:00
|
|
|
break;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (client->groupTransient()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
group = client->group();
|
|
|
|
break;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
client = client->transientFor();
|
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
// Use stacking order only to reduce flicker, it doesn't matter if block_stacking_updates == 0,
|
|
|
|
// I.e. if it's not up to date
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// SELI TODO: But maybe it should - what if a new client has been added that's not in stacking order yet?
|
2007-04-29 17:35:43 +00:00
|
|
|
ClientList to_show, to_hide;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = stacking_order.constBegin();
|
|
|
|
it != stacking_order.constEnd();
|
|
|
|
++it) {
|
|
|
|
if ((*it)->isUtility() || (*it)->isMenu() || (*it)->isToolbar()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
bool show = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!(*it)->isTransient()) {
|
|
|
|
if ((*it)->group()->members().count() == 1) // Has its own group, keep always visible
|
2007-04-29 17:35:43 +00:00
|
|
|
show = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (client != NULL && (*it)->group() == client->group())
|
2007-04-29 17:35:43 +00:00
|
|
|
show = true;
|
|
|
|
else
|
|
|
|
show = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
|
|
|
if (group != NULL && (*it)->group() == group)
|
2007-04-29 17:35:43 +00:00
|
|
|
show = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (client != NULL && client->hasTransient((*it), true))
|
2007-04-29 17:35:43 +00:00
|
|
|
show = true;
|
|
|
|
else
|
|
|
|
show = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (!show && also_hide) {
|
2007-04-29 17:35:43 +00:00
|
|
|
const ClientList mainclients = (*it)->mainClients();
|
2008-12-18 13:50:57 +00:00
|
|
|
// Don't hide utility windows which are standalone(?) or
|
2007-04-29 17:35:43 +00:00
|
|
|
// have e.g. kicker as mainwindow
|
2011-01-30 14:34:42 +00:00
|
|
|
if (mainclients.isEmpty())
|
2007-04-29 17:35:43 +00:00
|
|
|
show = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it2 = mainclients.constBegin();
|
|
|
|
it2 != mainclients.constEnd();
|
|
|
|
++it2) {
|
|
|
|
if ((*it2)->isSpecialWindow())
|
2007-04-29 17:35:43 +00:00
|
|
|
show = true;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!show)
|
|
|
|
to_hide.append(*it);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
if (show)
|
|
|
|
to_show.append(*it);
|
|
|
|
}
|
|
|
|
} // First show new ones, then hide
|
|
|
|
for (int i = to_show.size() - 1;
|
|
|
|
i >= 0;
|
|
|
|
--i) // From topmost
|
2008-12-18 13:50:57 +00:00
|
|
|
// TODO: Since this is in stacking order, the order of taskbar entries changes :(
|
2011-01-30 14:34:42 +00:00
|
|
|
to_show.at(i)->hideClient(false);
|
|
|
|
if (also_hide) {
|
|
|
|
for (ClientList::ConstIterator it = to_hide.constBegin();
|
|
|
|
it != to_hide.constEnd();
|
|
|
|
++it) // From bottommost
|
|
|
|
(*it)->hideClient(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
updateToolWindowsTimer.stop();
|
2011-01-30 14:34:42 +00:00
|
|
|
} else // setActiveClient() is after called with NULL client, quickly followed
|
|
|
|
// by setting a new client, which would result in flickering
|
2010-12-31 13:44:17 +00:00
|
|
|
resetUpdateToolWindowsTimer();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-12-31 13:44:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
void Workspace::resetUpdateToolWindowsTimer()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
updateToolWindowsTimer.start(200);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::slotUpdateToolWindows()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
updateToolWindows(true);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Updates the current colormap according to the currently active client
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
|
|
|
void Workspace::updateColormap()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
Colormap cmap = default_colormap;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (activeClient() && activeClient()->colormap() != None)
|
2007-04-29 17:35:43 +00:00
|
|
|
cmap = activeClient()->colormap();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (cmap != installed_colormap) {
|
|
|
|
XInstallColormap(display(), cmap);
|
2007-04-29 17:35:43 +00:00
|
|
|
installed_colormap = cmap;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-01-30 16:08:23 +00:00
|
|
|
void Workspace::slotReloadConfig()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2008-12-18 13:50:57 +00:00
|
|
|
reconfigure();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-01-30 16:08:23 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::reconfigure()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
reconfigureTimer.start(200);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* This D-Bus call is used by the compositing kcm. Since the reconfigure()
|
|
|
|
* D-Bus call delays the actual reconfiguring, it is not possible to immediately
|
|
|
|
* call compositingActive(). Therefore the kcm will instead call this to ensure
|
|
|
|
* the reconfiguring has already happened.
|
|
|
|
*/
|
2008-12-08 05:08:31 +00:00
|
|
|
bool Workspace::waitForCompositingSetup()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (reconfigureTimer.isActive()) {
|
2009-04-28 16:20:26 +00:00
|
|
|
reconfigureTimer.stop();
|
|
|
|
slotReconfigure();
|
2008-12-08 05:08:31 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
return compositingActive();
|
|
|
|
}
|
2008-12-08 05:08:31 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::slotSettingsChanged(int category)
|
|
|
|
{
|
|
|
|
kDebug(1212) << "Workspace::slotSettingsChanged()";
|
|
|
|
if (category == KGlobalSettings::SETTINGS_SHORTCUTS)
|
2007-04-29 17:35:43 +00:00
|
|
|
readShortcuts();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Reread settings
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
KWIN_PROCEDURE(CheckBorderSizesProcedure, Client, cl->checkBorderSizes(true));
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::slotReconfigure()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
kDebug(1212) << "Workspace::slotReconfigure()";
|
2007-04-29 17:35:43 +00:00
|
|
|
reconfigureTimer.stop();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
reserveElectricBorderActions(false);
|
|
|
|
if (options->electricBorders() == Options::ElectricAlways)
|
|
|
|
reserveElectricBorderSwitching(false);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2010-10-26 14:43:29 +00:00
|
|
|
bool borderlessMaximizedWindows = options->borderlessMaximizedWindows();
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
KGlobal::config()->reparseConfiguration();
|
|
|
|
unsigned long changed = options->updateSettings();
|
2008-11-25 01:45:09 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
tab_box->reconfigure();
|
2009-04-06 16:09:34 +00:00
|
|
|
desktop_change_osd->reconfigure();
|
2011-01-30 14:34:42 +00:00
|
|
|
initPositioning->reinitCascading(0);
|
2007-04-29 17:35:43 +00:00
|
|
|
readShortcuts();
|
2011-01-30 14:34:42 +00:00
|
|
|
forEachClient(CheckIgnoreFocusStealingProcedure());
|
|
|
|
updateToolWindows(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (mgr->reset(changed)) {
|
|
|
|
// Decorations need to be recreated
|
2008-12-18 13:50:57 +00:00
|
|
|
|
|
|
|
// This actually seems to make things worse now
|
|
|
|
//QWidget curtain;
|
|
|
|
//curtain.setBackgroundMode( NoBackground );
|
|
|
|
//curtain.setGeometry( Kephal::ScreenUtils::desktopGeometry() );
|
|
|
|
//curtain.show();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = clients.constBegin();
|
|
|
|
it != clients.constEnd();
|
|
|
|
++it)
|
|
|
|
(*it)->updateDecoration(true, true);
|
2009-11-15 03:24:04 +00:00
|
|
|
// If the new decoration doesn't supports tabs then ungroup clients
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!decorationSupportsClientGrouping()) {
|
2009-11-15 03:24:04 +00:00
|
|
|
QList<ClientGroup*> tmpGroups = clientGroups; // Prevent crashing
|
2011-01-30 14:34:42 +00:00
|
|
|
for (QList<ClientGroup*>::const_iterator i = tmpGroups.constBegin(); i != tmpGroups.constEnd(); i++)
|
2009-11-15 03:24:04 +00:00
|
|
|
(*i)->removeAll();
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
mgr->destroyPreviousPlugin();
|
|
|
|
} else {
|
|
|
|
forEachClient(CheckBorderSizesProcedure());
|
|
|
|
foreach (Client * c, clients)
|
|
|
|
c->triggerDecorationRepaint();
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
reserveElectricBorderActions(true);
|
|
|
|
if (options->electricBorders() == Options::ElectricAlways)
|
|
|
|
reserveElectricBorderSwitching(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
updateElectricBorders();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (options->topMenuEnabled() && !managingTopMenus()) {
|
|
|
|
if (topmenu_selection->claim(false))
|
2007-04-29 17:35:43 +00:00
|
|
|
setupTopMenuHandling();
|
|
|
|
else
|
|
|
|
lostTopMenuSelection();
|
2011-01-30 14:34:42 +00:00
|
|
|
} else if (!options->topMenuEnabled() && managingTopMenus()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
topmenu_selection->release();
|
|
|
|
lostTopMenuSelection();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
topmenu_height = 0; // Invalidate used menu height
|
2011-01-30 14:34:42 +00:00
|
|
|
if (managingTopMenus()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
updateTopMenuGeometry();
|
|
|
|
updateCurrentTopMenu();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-05-28 11:24:21 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (options->useCompositing && !compositingSuspended) {
|
2007-04-29 17:35:43 +00:00
|
|
|
setupCompositing();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (effects) // setupCompositing() may fail
|
2007-06-03 19:48:27 +00:00
|
|
|
effects->reconfigure();
|
2008-12-08 05:08:31 +00:00
|
|
|
addRepaintFull();
|
2011-01-30 14:34:42 +00:00
|
|
|
} else
|
2007-04-29 17:35:43 +00:00
|
|
|
finishCompositing();
|
|
|
|
|
|
|
|
loadWindowRules();
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::Iterator it = clients.begin();
|
|
|
|
it != clients.end();
|
|
|
|
++it) {
|
|
|
|
(*it)->setupWindowRules(true);
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->applyWindowRules();
|
2011-01-30 14:34:42 +00:00
|
|
|
discardUsedWindowRules(*it, false);
|
|
|
|
}
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (borderlessMaximizedWindows != options->borderlessMaximizedWindows() &&
|
|
|
|
!options->borderlessMaximizedWindows()) {
|
2010-10-26 14:43:29 +00:00
|
|
|
// in case borderless maximized windows option changed and new option
|
|
|
|
// is to have borders, we need to unset the borders for all maximized windows
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::Iterator it = clients.begin();
|
|
|
|
it != clients.end();
|
|
|
|
++it) {
|
|
|
|
if ((*it)->maximizeMode() == MaximizeFull)
|
2010-10-26 14:43:29 +00:00
|
|
|
(*it)->checkNoBorder();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-10-26 14:43:29 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
setTilingEnabled(options->tilingOn);
|
|
|
|
foreach (TilingLayout * layout, tilingLayouts) {
|
|
|
|
if (layout)
|
|
|
|
layout->reconfigureTiling();
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
// just so that we reset windows in the right manner, 'activate' the current active window
|
|
|
|
notifyTilingWindowActivated(activeClient());
|
|
|
|
rootInfo->setSupported(NET::WM2FrameOverlap, mgr->factory()->supports(AbilityExtendIntoClientArea));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-08 05:08:31 +00:00
|
|
|
void Workspace::slotReinitCompositing()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2008-12-08 05:08:31 +00:00
|
|
|
// Reparse config. Config options will be reloaded by setupCompositing()
|
2007-09-20 17:19:19 +00:00
|
|
|
KGlobal::config()->reparseConfiguration();
|
2007-09-25 18:32:26 +00:00
|
|
|
options->updateSettings();
|
2007-09-20 17:19:19 +00:00
|
|
|
|
2008-11-25 01:42:30 +00:00
|
|
|
// Update any settings that can be set in the compositing kcm.
|
|
|
|
updateElectricBorders();
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Restart compositing
|
2007-09-20 17:19:19 +00:00
|
|
|
finishCompositing();
|
2011-01-30 14:34:42 +00:00
|
|
|
|
2009-04-30 12:51:55 +00:00
|
|
|
// resume compositing if suspended
|
|
|
|
compositingSuspended = false;
|
2007-09-20 17:19:19 +00:00
|
|
|
setupCompositing();
|
2009-09-10 05:06:53 +00:00
|
|
|
KDecorationFactory* factory = mgr->factory();
|
|
|
|
factory->reset(SettingCompositing);
|
2011-01-30 14:34:42 +00:00
|
|
|
|
|
|
|
if (effects) { // setupCompositing() may fail
|
2007-09-20 17:19:19 +00:00
|
|
|
effects->reconfigure();
|
2011-01-30 14:34:42 +00:00
|
|
|
emit compositingToggled(true);
|
2007-09-20 17:19:19 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-09-20 17:19:19 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::loadDesktopSettings()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
KSharedConfig::Ptr c = KGlobal::config();
|
|
|
|
QString groupname;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (screen_number == 0)
|
2007-04-29 17:35:43 +00:00
|
|
|
groupname = "Desktops";
|
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
groupname.sprintf("Desktops-screen-%d", screen_number);
|
|
|
|
KConfigGroup group(c, groupname);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
int n = group.readEntry("Number", 4);
|
2009-02-14 15:40:52 +00:00
|
|
|
desktopCount_ = n;
|
2008-04-18 12:57:16 +00:00
|
|
|
workarea.clear();
|
2011-01-30 14:34:42 +00:00
|
|
|
workarea.resize(n + 1);
|
2009-02-17 15:50:00 +00:00
|
|
|
restrictedmovearea.clear();
|
2011-01-30 14:34:42 +00:00
|
|
|
restrictedmovearea.resize(n + 1);
|
2009-02-17 15:50:00 +00:00
|
|
|
oldrestrictedmovearea.clear();
|
2011-01-30 14:34:42 +00:00
|
|
|
oldrestrictedmovearea.resize(n + 1);
|
2008-04-18 12:57:16 +00:00
|
|
|
screenarea.clear();
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setNumberOfDesktops(n);
|
|
|
|
desktop_focus_chain.resize(n);
|
2008-12-18 13:50:57 +00:00
|
|
|
// Make it +1, so that it can be accessed as [1..numberofdesktops]
|
2011-01-30 14:34:42 +00:00
|
|
|
focus_chain.resize(n + 1);
|
|
|
|
for (int i = 1; i <= n; i++) {
|
|
|
|
QString s = group.readEntry(QString("Name_%1").arg(i), i18n("Desktop %1", i));
|
|
|
|
rootInfo->setDesktopName(i, s.toUtf8().data());
|
2007-04-29 17:35:43 +00:00
|
|
|
desktop_focus_chain[i-1] = i;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::saveDesktopSettings()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
KSharedConfig::Ptr c = KGlobal::config();
|
|
|
|
QString groupname;
|
|
|
|
if (screen_number == 0)
|
|
|
|
groupname = "Desktops";
|
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
groupname.sprintf("Desktops-screen-%d", screen_number);
|
|
|
|
KConfigGroup group(c, groupname);
|
|
|
|
|
|
|
|
group.writeEntry("Number", numberOfDesktops());
|
|
|
|
for (int i = 1; i <= numberOfDesktops(); i++) {
|
|
|
|
QString s = desktopName(i);
|
|
|
|
QString defaultvalue = i18n("Desktop %1", i);
|
|
|
|
if (s.isEmpty()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
s = defaultvalue;
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setDesktopName(i, s.toUtf8().data());
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (s != defaultvalue) {
|
|
|
|
group.writeEntry(QString("Name_%1").arg(i), s);
|
|
|
|
} else {
|
|
|
|
QString currentvalue = group.readEntry(QString("Name_%1").arg(i), QString());
|
|
|
|
if (currentvalue != defaultvalue)
|
|
|
|
group.writeEntry(QString("Name_%1").arg(i), "");
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-12-14 11:09:32 +00:00
|
|
|
|
|
|
|
// Save to disk
|
|
|
|
group.sync();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
QStringList Workspace::configModules(bool controlCenter)
|
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
QStringList args;
|
2007-09-06 20:34:44 +00:00
|
|
|
args << "kwindecoration";
|
2011-01-30 14:34:42 +00:00
|
|
|
if (controlCenter)
|
2007-09-06 20:34:44 +00:00
|
|
|
args << "kwinoptions";
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (KAuthorized::authorizeControlModule("kde-kwinoptions.desktop"))
|
2008-12-18 13:50:57 +00:00
|
|
|
args << "kwinactions" << "kwinfocus" << "kwinmoving" << "kwinadvanced"
|
2010-01-03 15:43:18 +00:00
|
|
|
<< "kwinrules" << "kwincompositing" << "kwintabbox" << "kwinscreenedges";
|
2007-04-29 17:35:43 +00:00
|
|
|
return args;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::configureWM()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2008-12-18 13:50:57 +00:00
|
|
|
QStringList args;
|
2011-01-30 14:34:42 +00:00
|
|
|
args << "--icon" << "preferences-system-windows" << configModules(false);
|
|
|
|
KToolInvocation::kdeinitExec("kcmshell4", args);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Avoids managing a window with title \a title
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::doNotManage(const QString& title)
|
|
|
|
{
|
|
|
|
doNotManageList.append(title);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Hack for java applets
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
bool Workspace::isNotManaged(const QString& title)
|
|
|
|
{
|
|
|
|
for (QStringList::Iterator it = doNotManageList.begin(); it != doNotManageList.end(); ++it) {
|
|
|
|
QRegExp r((*it));
|
|
|
|
if (r.indexIn(title) != -1) {
|
|
|
|
doNotManageList.erase(it);
|
2007-04-29 17:35:43 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
return false;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Refreshes all the client windows
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
|
|
|
void Workspace::refresh()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
QWidget w(NULL, Qt::X11BypassWindowManagerHint);
|
|
|
|
w.setGeometry(Kephal::ScreenUtils::desktopGeometry());
|
2007-04-29 17:35:43 +00:00
|
|
|
w.show();
|
|
|
|
w.hide();
|
|
|
|
QApplication::flush();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* During virt. desktop switching, desktop areas covered by windows that are
|
|
|
|
* going to be hidden are first obscured by new windows with no background
|
|
|
|
* ( i.e. transparent ) placed right below the windows. These invisible windows
|
|
|
|
* are removed after the switch is complete.
|
|
|
|
* Reduces desktop ( wallpaper ) repaints during desktop switching
|
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
class ObscuringWindows
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
~ObscuringWindows();
|
|
|
|
void create(Client* c);
|
|
|
|
private:
|
|
|
|
QList<Window> obscuring_windows;
|
|
|
|
static QList<Window>* cached;
|
|
|
|
static unsigned int max_cache_size;
|
|
|
|
};
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
QList<Window>* ObscuringWindows::cached = 0;
|
|
|
|
unsigned int ObscuringWindows::max_cache_size = 0;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void ObscuringWindows::create(Client* c)
|
|
|
|
{
|
|
|
|
if (compositing())
|
2008-12-18 13:50:57 +00:00
|
|
|
return; // Not needed with compositing
|
2011-01-30 14:34:42 +00:00
|
|
|
if (cached == 0)
|
2007-04-29 17:35:43 +00:00
|
|
|
cached = new QList<Window>;
|
|
|
|
Window obs_win;
|
|
|
|
XWindowChanges chngs;
|
|
|
|
int mask = CWSibling | CWStackMode;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (cached->count() > 0) {
|
|
|
|
cached->removeAll(obs_win = cached->first());
|
2007-04-29 17:35:43 +00:00
|
|
|
chngs.x = c->x();
|
|
|
|
chngs.y = c->y();
|
|
|
|
chngs.width = c->width();
|
|
|
|
chngs.height = c->height();
|
|
|
|
mask |= CWX | CWY | CWWidth | CWHeight;
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
2007-04-29 17:35:43 +00:00
|
|
|
XSetWindowAttributes a;
|
|
|
|
a.background_pixmap = None;
|
|
|
|
a.override_redirect = True;
|
2011-01-30 14:34:42 +00:00
|
|
|
obs_win = XCreateWindow(display(), rootWindow(), c->x(), c->y(),
|
|
|
|
c->width(), c->height(), 0, CopyFromParent, InputOutput,
|
|
|
|
CopyFromParent, CWBackPixmap | CWOverrideRedirect, &a);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
chngs.sibling = c->frameId();
|
|
|
|
chngs.stack_mode = Below;
|
2011-01-30 14:34:42 +00:00
|
|
|
XConfigureWindow(display(), obs_win, mask, &chngs);
|
|
|
|
XMapWindow(display(), obs_win);
|
|
|
|
obscuring_windows.append(obs_win);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
ObscuringWindows::~ObscuringWindows()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
max_cache_size = qMax(int(max_cache_size), obscuring_windows.count() + 4) - 1;
|
|
|
|
for (QList<Window>::ConstIterator it = obscuring_windows.constBegin();
|
|
|
|
it != obscuring_windows.constEnd();
|
|
|
|
++it) {
|
|
|
|
XUnmapWindow(display(), *it);
|
|
|
|
if (cached->count() < int(max_cache_size))
|
|
|
|
cached->prepend(*it);
|
2007-04-29 17:35:43 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
XDestroyWindow(display(), *it);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Sets the current desktop to \a new_desktop
|
|
|
|
*
|
|
|
|
* Shows/Hides windows according to the stacking order and finally
|
|
|
|
* propages the new desktop to the world
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
bool Workspace::setCurrentDesktop(int new_desktop)
|
|
|
|
{
|
|
|
|
if (new_desktop < 1 || new_desktop > numberOfDesktops())
|
2007-04-29 17:35:43 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
closeActivePopup();
|
|
|
|
++block_focus;
|
2008-12-18 13:50:57 +00:00
|
|
|
// TODO: Q_ASSERT( block_stacking_updates == 0 ); // Make sure stacking_order is up to date
|
2011-01-30 14:34:42 +00:00
|
|
|
StackingUpdatesBlocker blocker(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2009-02-14 09:46:12 +00:00
|
|
|
int old_desktop = currentDesktop();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (new_desktop != currentDesktop()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
++block_showing_desktop;
|
2008-12-18 13:50:57 +00:00
|
|
|
// Optimized Desktop switching: unmapping done from back to front
|
|
|
|
// mapping done from front to back => less exposure events
|
2011-01-30 14:34:42 +00:00
|
|
|
Notify::raise((Notify::Event)(Notify::DesktopChange + new_desktop));
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
ObscuringWindows obs_wins;
|
|
|
|
|
2009-02-14 15:40:52 +00:00
|
|
|
currentDesktop_ = new_desktop; // Change the desktop (so that Client::updateVisibility() works)
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = stacking_order.constBegin();
|
|
|
|
it != stacking_order.constEnd();
|
|
|
|
++it)
|
|
|
|
if (!(*it)->isOnDesktop(new_desktop) && (*it) != movingClient && (*it)->isOnCurrentActivity()) {
|
|
|
|
if ((*it)->isShown(true) && (*it)->isOnDesktop(old_desktop))
|
|
|
|
obs_wins.create(*it);
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->updateVisibility();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Now propagate the change, after hiding, before showing
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setCurrentDesktop(currentDesktop());
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2010-05-09 06:15:40 +00:00
|
|
|
// if the client is moved to another desktop, that desktop may
|
|
|
|
// not have an existing layout. In addition this tiling layout
|
|
|
|
// will require rearrangement, so notify about desktop changes.
|
2011-01-30 14:34:42 +00:00
|
|
|
if (movingClient && !movingClient->isOnDesktop(new_desktop)) {
|
2010-04-25 16:43:14 +00:00
|
|
|
int old_desktop = movingClient->desktop();
|
2011-01-30 14:34:42 +00:00
|
|
|
movingClient->setDesktop(new_desktop);
|
|
|
|
if (tilingEnabled()) {
|
|
|
|
notifyTilingWindowDesktopChanged(movingClient, old_desktop);
|
2010-04-25 16:43:14 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = stacking_order.size() - 1; i >= 0 ; --i)
|
|
|
|
if (stacking_order.at(i)->isOnDesktop(new_desktop) && stacking_order.at(i)->isOnCurrentActivity())
|
|
|
|
stacking_order.at(i)->updateVisibility();
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
--block_showing_desktop;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (showingDesktop()) // Do this only after desktop change to avoid flicker
|
|
|
|
resetShowingDesktop(false);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Restore the focus on this desktop
|
2007-04-29 17:35:43 +00:00
|
|
|
--block_focus;
|
|
|
|
Client* c = 0;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (options->focusPolicyIsReasonable()) {
|
|
|
|
// Search in focus chain
|
|
|
|
if (movingClient != NULL && active_client == movingClient &&
|
|
|
|
focus_chain[currentDesktop()].contains(active_client) &&
|
|
|
|
active_client->isShown(true) && active_client->isOnCurrentDesktop())
|
2008-12-18 13:50:57 +00:00
|
|
|
c = active_client; // The requestFocus below will fail, as the client is already active
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!c) {
|
|
|
|
for (int i = focus_chain[currentDesktop()].size() - 1; i >= 0; --i) {
|
|
|
|
if (focus_chain[currentDesktop()].at(i)->isShown(false) &&
|
|
|
|
focus_chain[currentDesktop()].at(i)->isOnCurrentActivity()) {
|
|
|
|
c = focus_chain[currentDesktop()].at(i);
|
2007-04-29 17:35:43 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
// If "unreasonable focus policy" and active_client is on_all_desktops and
|
|
|
|
// under mouse (Hence == old_active_client), conserve focus.
|
|
|
|
// (Thanks to Volker Schatz <V.Schatz at thphys.uni-heidelberg.de>)
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (active_client && active_client->isShown(true) && active_client->isOnCurrentDesktop())
|
2008-12-18 13:50:57 +00:00
|
|
|
c = active_client;
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c == NULL && !desktops.isEmpty())
|
|
|
|
c = findDesktop(true, currentDesktop());
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c != active_client)
|
|
|
|
setActiveClient(NULL, Allowed);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c)
|
|
|
|
requestFocus(c);
|
|
|
|
else if (!desktops.isEmpty())
|
|
|
|
requestFocus(findDesktop(true, currentDesktop()));
|
2007-04-29 17:35:43 +00:00
|
|
|
else
|
|
|
|
focusToNull();
|
|
|
|
|
|
|
|
updateCurrentTopMenu();
|
|
|
|
|
|
|
|
// Update focus chain:
|
|
|
|
// If input: chain = { 1, 2, 3, 4 } and currentDesktop() = 3,
|
|
|
|
// Output: chain = { 3, 1, 2, 4 }.
|
2008-12-18 13:50:57 +00:00
|
|
|
//kDebug(1212) << QString("Switching to desktop #%1, at focus_chain index %2\n")
|
|
|
|
// .arg(currentDesktop()).arg(desktop_focus_chain.find( currentDesktop() ));
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = desktop_focus_chain.indexOf(currentDesktop()); i > 0; i--)
|
2007-04-29 17:35:43 +00:00
|
|
|
desktop_focus_chain[i] = desktop_focus_chain[i-1];
|
|
|
|
desktop_focus_chain[0] = currentDesktop();
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
//QString s = "desktop_focus_chain[] = { ";
|
2011-01-30 14:34:42 +00:00
|
|
|
//for ( uint i = 0; i < desktop_focus_chain.size(); i++ )
|
2008-12-18 13:50:57 +00:00
|
|
|
// s += QString::number( desktop_focus_chain[i] ) + ", ";
|
|
|
|
//kDebug( 1212 ) << s << "}\n";
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2009-04-06 16:09:34 +00:00
|
|
|
// Not for the very first time, only if something changed and there are more than 1 desktops
|
2011-01-30 14:34:42 +00:00
|
|
|
if (old_desktop != 0 && old_desktop != new_desktop && numberOfDesktops() > 1)
|
|
|
|
desktop_change_osd->desktopChanged(old_desktop);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (effects != NULL && old_desktop != 0 && old_desktop != new_desktop)
|
|
|
|
static_cast<EffectsHandlerImpl*>(effects)->desktopChanged(old_desktop);
|
|
|
|
if (compositing())
|
2007-04-29 17:35:43 +00:00
|
|
|
addRepaintFull();
|
2011-01-30 14:34:42 +00:00
|
|
|
|
2010-09-21 14:31:40 +00:00
|
|
|
emit currentDesktopChanged(old_desktop);
|
2007-04-29 17:35:43 +00:00
|
|
|
return true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2010-05-11 20:30:20 +00:00
|
|
|
/**
|
|
|
|
* Updates the current activity when it changes
|
|
|
|
* do *not* call this directly; it does not set the activity.
|
|
|
|
*
|
|
|
|
* Shows/Hides windows according to the stacking order
|
|
|
|
*/
|
|
|
|
void Workspace::updateCurrentActivity(const QString &new_activity)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2010-05-11 20:30:20 +00:00
|
|
|
|
|
|
|
//closeActivePopup();
|
|
|
|
++block_focus;
|
|
|
|
// TODO: Q_ASSERT( block_stacking_updates == 0 ); // Make sure stacking_order is up to date
|
2011-01-30 14:34:42 +00:00
|
|
|
StackingUpdatesBlocker blocker(this);
|
2010-05-11 20:30:20 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (new_activity != activity_) {
|
2010-05-11 20:30:20 +00:00
|
|
|
++block_showing_desktop; //FIXME should I be using that?
|
|
|
|
// Optimized Desktop switching: unmapping done from back to front
|
|
|
|
// mapping done from front to back => less exposure events
|
|
|
|
//Notify::raise((Notify::Event) (Notify::DesktopChange+new_desktop));
|
|
|
|
|
|
|
|
ObscuringWindows obs_wins;
|
|
|
|
|
|
|
|
QString old_activity = activity_;
|
|
|
|
activity_ = new_activity;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = stacking_order.constBegin();
|
|
|
|
it != stacking_order.constEnd();
|
|
|
|
++it)
|
|
|
|
if (!(*it)->isOnActivity(new_activity) && (*it) != movingClient && (*it)->isOnCurrentDesktop()) {
|
|
|
|
if ((*it)->isShown(true) && (*it)->isOnActivity(old_activity))
|
|
|
|
obs_wins.create(*it);
|
2010-05-11 20:30:20 +00:00
|
|
|
(*it)->updateVisibility();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-05-11 20:30:20 +00:00
|
|
|
|
|
|
|
// Now propagate the change, after hiding, before showing
|
|
|
|
//rootInfo->setCurrentDesktop( currentDesktop() );
|
|
|
|
|
|
|
|
/* TODO someday enable dragging windows to other activities
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( movingClient && !movingClient->isOnDesktop( new_desktop ))
|
2010-05-11 20:30:20 +00:00
|
|
|
{
|
|
|
|
int old_desktop = movingClient->desktop();
|
|
|
|
movingClient->setDesktop( new_desktop );
|
2011-01-30 14:34:42 +00:00
|
|
|
if ( tilingEnabled() )
|
2010-05-11 20:30:20 +00:00
|
|
|
{
|
|
|
|
notifyWindowDesktopChanged( movingClient, old_desktop );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = stacking_order.size() - 1; i >= 0 ; --i)
|
|
|
|
if (stacking_order.at(i)->isOnActivity(new_activity))
|
|
|
|
stacking_order.at(i)->updateVisibility();
|
2010-05-11 20:30:20 +00:00
|
|
|
|
|
|
|
--block_showing_desktop;
|
|
|
|
//FIXME not sure if I should do this either
|
2011-01-30 14:34:42 +00:00
|
|
|
if (showingDesktop()) // Do this only after desktop change to avoid flicker
|
|
|
|
resetShowingDesktop(false);
|
|
|
|
}
|
2010-05-11 20:30:20 +00:00
|
|
|
|
|
|
|
// Restore the focus on this desktop
|
|
|
|
--block_focus;
|
|
|
|
Client* c = 0;
|
|
|
|
|
|
|
|
//FIXME below here is a lot of focuschain stuff, probably all wrong now
|
2011-01-30 14:34:42 +00:00
|
|
|
if (options->focusPolicyIsReasonable()) {
|
|
|
|
// Search in focus chain
|
|
|
|
if (movingClient != NULL && active_client == movingClient &&
|
|
|
|
focus_chain[currentDesktop()].contains(active_client) &&
|
|
|
|
active_client->isShown(true) && active_client->isOnCurrentDesktop())
|
2010-05-11 20:30:20 +00:00
|
|
|
c = active_client; // The requestFocus below will fail, as the client is already active
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!c) {
|
|
|
|
for (int i = focus_chain[currentDesktop()].size() - 1; i >= 0; --i) {
|
|
|
|
if (focus_chain[currentDesktop()].at(i)->isShown(false) &&
|
|
|
|
focus_chain[currentDesktop()].at(i)->isOnCurrentActivity()) {
|
|
|
|
c = focus_chain[currentDesktop()].at(i);
|
2010-05-11 20:30:20 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-05-11 20:30:20 +00:00
|
|
|
// If "unreasonable focus policy" and active_client is on_all_desktops and
|
|
|
|
// under mouse (Hence == old_active_client), conserve focus.
|
|
|
|
// (Thanks to Volker Schatz <V.Schatz at thphys.uni-heidelberg.de>)
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (active_client && active_client->isShown(true) && active_client->isOnCurrentDesktop() && active_client->isOnCurrentActivity())
|
2010-05-11 20:30:20 +00:00
|
|
|
c = active_client;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c == NULL && !desktops.isEmpty())
|
|
|
|
c = findDesktop(true, currentDesktop());
|
2010-05-11 20:30:20 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c != active_client)
|
|
|
|
setActiveClient(NULL, Allowed);
|
2010-05-11 20:30:20 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c)
|
|
|
|
requestFocus(c);
|
|
|
|
else if (!desktops.isEmpty())
|
|
|
|
requestFocus(findDesktop(true, currentDesktop()));
|
2010-05-11 20:30:20 +00:00
|
|
|
else
|
|
|
|
focusToNull();
|
|
|
|
|
|
|
|
updateCurrentTopMenu();
|
|
|
|
|
|
|
|
// Update focus chain:
|
|
|
|
// If input: chain = { 1, 2, 3, 4 } and currentDesktop() = 3,
|
|
|
|
// Output: chain = { 3, 1, 2, 4 }.
|
|
|
|
//kDebug(1212) << QString("Switching to desktop #%1, at focus_chain index %2\n")
|
|
|
|
// .arg(currentDesktop()).arg(desktop_focus_chain.find( currentDesktop() ));
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = desktop_focus_chain.indexOf(currentDesktop()); i > 0; i--)
|
2010-05-11 20:30:20 +00:00
|
|
|
desktop_focus_chain[i] = desktop_focus_chain[i-1];
|
|
|
|
desktop_focus_chain[0] = currentDesktop();
|
|
|
|
|
|
|
|
//QString s = "desktop_focus_chain[] = { ";
|
2011-01-30 14:34:42 +00:00
|
|
|
//for ( uint i = 0; i < desktop_focus_chain.size(); i++ )
|
2010-05-11 20:30:20 +00:00
|
|
|
// s += QString::number( desktop_focus_chain[i] ) + ", ";
|
|
|
|
//kDebug( 1212 ) << s << "}\n";
|
|
|
|
|
|
|
|
// Not for the very first time, only if something changed and there are more than 1 desktops
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
//if ( effects != NULL && old_desktop != 0 && old_desktop != new_desktop )
|
2010-05-11 20:30:20 +00:00
|
|
|
// static_cast<EffectsHandlerImpl*>( effects )->desktopChanged( old_desktop );
|
2011-01-30 14:34:42 +00:00
|
|
|
if (compositing())
|
2010-05-11 20:30:20 +00:00
|
|
|
addRepaintFull();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-05-11 20:30:20 +00:00
|
|
|
|
2010-05-13 04:06:27 +00:00
|
|
|
/**
|
|
|
|
* updates clients when an activity is destroyed.
|
|
|
|
* this ensures that a client does not get 'lost' if the only activity it's on is removed.
|
|
|
|
*/
|
|
|
|
void Workspace::activityRemoved(const QString &activity)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2010-09-24 12:03:22 +00:00
|
|
|
allActivities_.removeOne(activity);
|
2011-01-30 14:34:42 +00:00
|
|
|
foreach (Client * client, stacking_order) {
|
2010-05-13 04:06:27 +00:00
|
|
|
client->setOnActivity(activity, false);
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-11-01 19:53:20 +00:00
|
|
|
//toss out any session data for it
|
|
|
|
KConfigGroup cg(KGlobal::config(), QString("SubSession: ") + activity);
|
|
|
|
cg.deleteGroup();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-05-13 04:06:27 +00:00
|
|
|
|
2010-09-24 12:03:22 +00:00
|
|
|
void Workspace::activityAdded(const QString &activity)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2010-09-24 12:03:22 +00:00
|
|
|
allActivities_ << activity;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-09-24 12:03:22 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Called only from D-Bus
|
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::nextDesktop()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
int desktop = currentDesktop() + 1;
|
2011-01-30 14:34:42 +00:00
|
|
|
setCurrentDesktop(desktop > numberOfDesktops() ? 1 : desktop);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Called only from D-Bus
|
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::previousDesktop()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
int desktop = currentDesktop() - 1;
|
2011-01-30 14:34:42 +00:00
|
|
|
setCurrentDesktop(desktop > 0 ? desktop : numberOfDesktops());
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Sets the number of virtual desktops to \a n
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::setNumberOfDesktops(int n)
|
|
|
|
{
|
|
|
|
if (n > KWIN_MAX_NUMBER_DESKTOPS)
|
2011-01-13 19:05:28 +00:00
|
|
|
n = KWIN_MAX_NUMBER_DESKTOPS;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (n < 1 || n == numberOfDesktops())
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2009-02-14 09:46:12 +00:00
|
|
|
int old_number_of_desktops = numberOfDesktops();
|
2009-02-14 15:40:52 +00:00
|
|
|
desktopCount_ = n;
|
|
|
|
updateDesktopLayout(); // Make sure the layout is still valid
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (currentDesktop() > numberOfDesktops())
|
|
|
|
setCurrentDesktop(numberOfDesktops());
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// If increasing the number, do the resizing now, otherwise
|
|
|
|
// after the moving of windows to still existing desktops
|
2011-01-30 14:34:42 +00:00
|
|
|
if (old_number_of_desktops < numberOfDesktops()) {
|
|
|
|
rootInfo->setNumberOfDesktops(numberOfDesktops());
|
2009-02-14 09:46:12 +00:00
|
|
|
NETPoint* viewports = new NETPoint[numberOfDesktops()];
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setDesktopViewport(numberOfDesktops(), *viewports);
|
2007-04-29 17:35:43 +00:00
|
|
|
delete[] viewports;
|
2011-01-30 14:34:42 +00:00
|
|
|
updateClientArea(true);
|
|
|
|
focus_chain.resize(numberOfDesktops() + 1);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// If the number of desktops decreased, move all windows
|
|
|
|
// that would be hidden to the last visible desktop
|
2011-01-30 14:34:42 +00:00
|
|
|
if (old_number_of_desktops > numberOfDesktops()) {
|
|
|
|
for (ClientList::ConstIterator it = clients.constBegin();
|
|
|
|
it != clients.constEnd();
|
|
|
|
++it)
|
|
|
|
if (!(*it)->isOnAllDesktops() && (*it)->desktop() > numberOfDesktops())
|
|
|
|
sendClientToDesktop(*it, numberOfDesktops(), true);
|
2010-04-25 16:43:14 +00:00
|
|
|
// TODO: Tile should have a method allClients, push them into other tiles
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (old_number_of_desktops > numberOfDesktops()) {
|
|
|
|
rootInfo->setNumberOfDesktops(numberOfDesktops());
|
2009-02-14 09:46:12 +00:00
|
|
|
NETPoint* viewports = new NETPoint[numberOfDesktops()];
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setDesktopViewport(numberOfDesktops(), *viewports);
|
2007-04-29 17:35:43 +00:00
|
|
|
delete[] viewports;
|
2011-01-30 14:34:42 +00:00
|
|
|
updateClientArea(true);
|
|
|
|
focus_chain.resize(numberOfDesktops() + 1);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
saveDesktopSettings();
|
|
|
|
|
|
|
|
// Resize and reset the desktop focus chain.
|
2011-01-30 14:34:42 +00:00
|
|
|
desktop_focus_chain.resize(n);
|
|
|
|
for (int i = 0; i < int(desktop_focus_chain.size()); i++)
|
|
|
|
desktop_focus_chain[i] = i + 1;
|
2009-04-06 16:09:34 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
tilingLayouts.resize(numberOfDesktops() + 1);
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2009-04-06 16:09:34 +00:00
|
|
|
// reset the desktop change osd
|
|
|
|
desktop_change_osd->numberDesktopsChanged();
|
2010-03-28 09:04:14 +00:00
|
|
|
// inform effects
|
2011-01-30 14:34:42 +00:00
|
|
|
if (effects)
|
|
|
|
static_cast< EffectsHandlerImpl* >(effects)->numberDesktopsChanged(old_number_of_desktops);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Sends client \a c to desktop \a desk.
|
|
|
|
*
|
|
|
|
* Takes care of transients as well.
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::sendClientToDesktop(Client* c, int desk, bool dont_activate)
|
|
|
|
{
|
2010-04-25 16:43:14 +00:00
|
|
|
int old_desktop = c->desktop();
|
2011-01-30 14:34:42 +00:00
|
|
|
bool was_on_desktop = c->isOnDesktop(desk) || c->isOnAllDesktops();
|
|
|
|
c->setDesktop(desk);
|
|
|
|
if (c->desktop() != desk) // No change or desktop forced
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
|
|
|
desk = c->desktop(); // Client did range checking
|
2011-01-30 14:34:42 +00:00
|
|
|
|
2010-09-21 14:31:40 +00:00
|
|
|
emit desktopPresenceChanged(c, old_desktop);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c->isOnDesktop(currentDesktop())) {
|
|
|
|
if (c->wantsTabFocus() && options->focusPolicyIsReasonable() &&
|
|
|
|
!was_on_desktop && // for stickyness changes
|
|
|
|
!dont_activate)
|
|
|
|
requestFocus(c);
|
2007-04-29 17:35:43 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
restackClientUnderActive(c);
|
|
|
|
} else
|
|
|
|
raiseClient(c);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
notifyTilingWindowDesktopChanged(c, old_desktop);
|
2010-04-25 16:43:14 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
ClientList transients_stacking_order = ensureStackingOrder(c->transients());
|
|
|
|
for (ClientList::ConstIterator it = transients_stacking_order.constBegin();
|
|
|
|
it != transients_stacking_order.constEnd();
|
|
|
|
++it)
|
|
|
|
sendClientToDesktop(*it, desk, dont_activate);
|
2007-04-29 17:35:43 +00:00
|
|
|
updateClientArea();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2010-05-11 20:30:20 +00:00
|
|
|
/**
|
2010-05-19 21:02:40 +00:00
|
|
|
* Adds/removes client \a c to/from \a activity.
|
2010-05-11 20:30:20 +00:00
|
|
|
*
|
|
|
|
* Takes care of transients as well.
|
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::toggleClientOnActivity(Client* c, const QString &activity, bool dont_activate)
|
|
|
|
{
|
2010-05-11 20:30:20 +00:00
|
|
|
//int old_desktop = c->desktop();
|
|
|
|
bool was_on_activity = c->isOnActivity(activity);
|
|
|
|
bool was_on_all = c->isOnAllActivities();
|
|
|
|
//note: all activities === no activities
|
|
|
|
bool enable = was_on_all || !was_on_activity;
|
2011-01-30 14:34:42 +00:00
|
|
|
c->setOnActivity(activity, enable);
|
|
|
|
if (c->isOnActivity(activity) == was_on_activity && c->isOnAllActivities() == was_on_all) // No change
|
2010-05-11 20:30:20 +00:00
|
|
|
return;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c->isOnCurrentActivity()) {
|
|
|
|
if (c->wantsTabFocus() && options->focusPolicyIsReasonable() &&
|
|
|
|
!was_on_activity && // for stickyness changes
|
|
|
|
//FIXME not sure if the line above refers to the correct activity
|
|
|
|
!dont_activate)
|
|
|
|
requestFocus(c);
|
2010-05-11 20:30:20 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
restackClientUnderActive(c);
|
|
|
|
} else
|
|
|
|
raiseClient(c);
|
2010-05-11 20:30:20 +00:00
|
|
|
|
|
|
|
//notifyWindowDesktopChanged( c, old_desktop );
|
|
|
|
//FIXME does tiling break?
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
ClientList transients_stacking_order = ensureStackingOrder(c->transients());
|
|
|
|
for (ClientList::ConstIterator it = transients_stacking_order.constBegin();
|
|
|
|
it != transients_stacking_order.constEnd();
|
|
|
|
++it)
|
|
|
|
toggleClientOnActivity(*it, activity, dont_activate);
|
2010-05-11 20:30:20 +00:00
|
|
|
updateClientArea();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2010-05-11 20:30:20 +00:00
|
|
|
|
2007-05-07 12:18:19 +00:00
|
|
|
int Workspace::numScreens() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (!options->xineramaEnabled)
|
2007-05-17 17:25:08 +00:00
|
|
|
return 1;
|
2008-11-15 03:04:44 +00:00
|
|
|
return Kephal::ScreenUtils::numScreens();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-05-07 12:18:19 +00:00
|
|
|
|
2009-02-21 08:33:13 +00:00
|
|
|
int Workspace::activeScreen() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (!options->xineramaEnabled)
|
2007-05-07 12:18:19 +00:00
|
|
|
return 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!options->activeMouseScreen) {
|
|
|
|
if (activeClient() != NULL && !activeClient()->isOnScreen(active_screen))
|
2009-02-18 04:05:37 +00:00
|
|
|
return activeClient()->screen();
|
|
|
|
return active_screen;
|
2007-05-07 12:18:19 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
return Kephal::ScreenUtils::screenId(cursorPos());
|
|
|
|
}
|
2007-05-07 12:18:19 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Check whether a client moved completely out of what's considered the active screen,
|
|
|
|
* if yes, set a new active screen.
|
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::checkActiveScreen(const Client* c)
|
|
|
|
{
|
|
|
|
if (!options->xineramaEnabled)
|
2007-05-09 12:26:32 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!c->isActive())
|
2007-05-09 12:26:32 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!c->isOnScreen(active_screen))
|
2007-05-09 12:26:32 +00:00
|
|
|
active_screen = c->screen();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-05-09 12:26:32 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Called e.g. when a user clicks on a window, set active screen to be the screen
|
|
|
|
* where the click occurred
|
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::setActiveScreenMouse(const QPoint& mousepos)
|
|
|
|
{
|
|
|
|
if (!options->xineramaEnabled)
|
2007-05-09 12:30:29 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
active_screen = Kephal::ScreenUtils::screenId(mousepos);
|
|
|
|
}
|
2007-05-09 12:30:29 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
QRect Workspace::screenGeometry(int screen) const
|
|
|
|
{
|
|
|
|
if (!options->xineramaEnabled)
|
2008-11-15 03:04:44 +00:00
|
|
|
return Kephal::ScreenUtils::desktopGeometry();
|
2011-01-30 14:34:42 +00:00
|
|
|
return Kephal::ScreenUtils::screenGeometry(screen);
|
|
|
|
}
|
2007-05-07 12:18:19 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
int Workspace::screenNumber(const QPoint& pos) const
|
|
|
|
{
|
|
|
|
if (!options->xineramaEnabled)
|
2007-05-07 12:18:19 +00:00
|
|
|
return 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
return Kephal::ScreenUtils::screenId(pos);
|
|
|
|
}
|
2007-05-07 12:18:19 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::sendClientToScreen(Client* c, int screen)
|
|
|
|
{
|
|
|
|
if (c->screen() == screen) // Don't use isOnScreen(), that's true even when only partially
|
2007-05-07 13:13:48 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
GeometryUpdatesBlocker blocker(c);
|
|
|
|
QRect old_sarea = clientArea(MaximizeArea, c);
|
|
|
|
QRect sarea = clientArea(MaximizeArea, screen, c->desktop());
|
|
|
|
c->setGeometry(sarea.x() - old_sarea.x() + c->x(), sarea.y() - old_sarea.y() + c->y(),
|
|
|
|
c->size().width(), c->size().height());
|
2007-05-07 13:13:48 +00:00
|
|
|
c->checkWorkspacePosition();
|
2011-01-30 14:34:42 +00:00
|
|
|
ClientList transients_stacking_order = ensureStackingOrder(c->transients());
|
|
|
|
for (ClientList::ConstIterator it = transients_stacking_order.constBegin();
|
|
|
|
it != transients_stacking_order.constEnd();
|
|
|
|
++it)
|
|
|
|
sendClientToScreen(*it, screen);
|
|
|
|
if (c->isActive())
|
2007-05-09 12:26:32 +00:00
|
|
|
active_screen = screen;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-05-07 13:13:48 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::killWindowId(Window window_to_kill)
|
|
|
|
{
|
|
|
|
if (window_to_kill == None)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
|
|
|
Window window = window_to_kill;
|
|
|
|
Client* client = NULL;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (;;) {
|
|
|
|
client = findClient(FrameIdMatchPredicate(window));
|
|
|
|
if (client != NULL)
|
2008-12-18 13:50:57 +00:00
|
|
|
break; // Found the client
|
2007-04-29 17:35:43 +00:00
|
|
|
Window parent, root;
|
|
|
|
Window* children;
|
|
|
|
unsigned int children_count;
|
2011-01-30 14:34:42 +00:00
|
|
|
XQueryTree(display(), window, &root, &parent, &children, &children_count);
|
|
|
|
if (children != NULL)
|
|
|
|
XFree(children);
|
|
|
|
if (window == root) // We didn't find the client, probably an override-redirect window
|
2007-04-29 17:35:43 +00:00
|
|
|
break;
|
2008-12-18 13:50:57 +00:00
|
|
|
window = parent; // Go up
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (client != NULL)
|
2007-04-29 17:35:43 +00:00
|
|
|
client->killWindow();
|
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
XKillClient(display(), window_to_kill);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::sendPingToWindow(Window window, Time timestamp)
|
|
|
|
{
|
|
|
|
rootInfo->sendPing(window, timestamp);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::sendTakeActivity(Client* c, Time timestamp, long flags)
|
|
|
|
{
|
|
|
|
rootInfo->takeActivity(c->window(), timestamp, flags);
|
2007-04-29 17:35:43 +00:00
|
|
|
pending_take_activity = c;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Invokes keyboard mouse emulation
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
|
|
|
void Workspace::slotMouseEmulation()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (mouse_emulation) {
|
2007-04-29 17:35:43 +00:00
|
|
|
ungrabXKeyboard();
|
|
|
|
mouse_emulation = false;
|
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (grabXKeyboard()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
mouse_emulation = true;
|
|
|
|
mouse_emulation_state = 0;
|
|
|
|
mouse_emulation_window = 0;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Returns the child window under the mouse and activates the
|
|
|
|
* respective client if necessary.
|
|
|
|
*
|
|
|
|
* Auxiliary function for the mouse emulation system.
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
|
|
|
WId Workspace::getMouseEmulationWindow()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
Window root;
|
|
|
|
Window child = rootWindow();
|
|
|
|
int root_x, root_y, lx, ly;
|
|
|
|
uint state;
|
|
|
|
Window w;
|
|
|
|
Client * c = 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
do {
|
2007-04-29 17:35:43 +00:00
|
|
|
w = child;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!c)
|
|
|
|
c = findClient(FrameIdMatchPredicate(w));
|
|
|
|
XQueryPointer(display(), w, &root, &child, &root_x, &root_y, &lx, &ly, &state);
|
|
|
|
} while (child != None && child != w);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c && !c->isActive())
|
|
|
|
activateClient(c);
|
|
|
|
return WId(w);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Sends a faked mouse event to the specified window. Returns the new button state.
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
unsigned int Workspace::sendFakedMouseEvent(const QPoint& pos, WId w, MouseEmulation type,
|
|
|
|
int button, unsigned int state)
|
|
|
|
{
|
|
|
|
if (!w)
|
2007-04-29 17:35:43 +00:00
|
|
|
return state;
|
2011-01-30 14:34:42 +00:00
|
|
|
QWidget* widget = QWidget::find(w);
|
|
|
|
if ((!widget || qobject_cast<QToolButton*>(widget)) && !findClient(WindowMatchPredicate(w))) {
|
2007-04-29 17:35:43 +00:00
|
|
|
int x, y;
|
|
|
|
Window xw;
|
2011-01-30 14:34:42 +00:00
|
|
|
XTranslateCoordinates(display(), rootWindow(), w, pos.x(), pos.y(), &x, &y, &xw);
|
|
|
|
if (type == EmuMove) {
|
|
|
|
// Motion notify events
|
2007-04-29 17:35:43 +00:00
|
|
|
XEvent e;
|
|
|
|
e.type = MotionNotify;
|
|
|
|
e.xmotion.window = w;
|
|
|
|
e.xmotion.root = rootWindow();
|
|
|
|
e.xmotion.subwindow = w;
|
|
|
|
e.xmotion.time = xTime();
|
|
|
|
e.xmotion.x = x;
|
|
|
|
e.xmotion.y = y;
|
|
|
|
e.xmotion.x_root = pos.x();
|
|
|
|
e.xmotion.y_root = pos.y();
|
|
|
|
e.xmotion.state = state;
|
|
|
|
e.xmotion.is_hint = NotifyNormal;
|
2011-01-30 14:34:42 +00:00
|
|
|
XSendEvent(display(), w, true, ButtonMotionMask, &e);
|
|
|
|
} else {
|
2007-04-29 17:35:43 +00:00
|
|
|
XEvent e;
|
|
|
|
e.type = type == EmuRelease ? ButtonRelease : ButtonPress;
|
|
|
|
e.xbutton.window = w;
|
|
|
|
e.xbutton.root = rootWindow();
|
|
|
|
e.xbutton.subwindow = w;
|
|
|
|
e.xbutton.time = xTime();
|
|
|
|
e.xbutton.x = x;
|
|
|
|
e.xbutton.y = y;
|
|
|
|
e.xbutton.x_root = pos.x();
|
|
|
|
e.xbutton.y_root = pos.y();
|
|
|
|
e.xbutton.state = state;
|
|
|
|
e.xbutton.button = button;
|
2011-01-30 14:34:42 +00:00
|
|
|
XSendEvent(display(), w, true, ButtonPressMask, &e);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (type == EmuPress) {
|
|
|
|
switch(button) {
|
|
|
|
case 2:
|
|
|
|
state |= Button2Mask;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
state |= Button3Mask;
|
|
|
|
break;
|
|
|
|
default: // 1
|
|
|
|
state |= Button1Mask;
|
|
|
|
break;
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
|
|
|
switch(button) {
|
|
|
|
case 2:
|
|
|
|
state &= ~Button2Mask;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
state &= ~Button3Mask;
|
|
|
|
break;
|
|
|
|
default: // 1
|
|
|
|
state &= ~Button1Mask;
|
|
|
|
break;
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
return state;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Handles keypress event during mouse emulation
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
bool Workspace::keyPressMouseEmulation(XKeyEvent& ev)
|
|
|
|
{
|
|
|
|
int kc = XKeycodeToKeysym(display(), ev.keycode, 0);
|
|
|
|
int km = ev.state & (ControlMask | Mod1Mask | ShiftMask);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
bool is_control = km & ControlMask;
|
|
|
|
bool is_alt = km & Mod1Mask;
|
|
|
|
bool is_shift = km & ShiftMask;
|
2011-01-30 14:34:42 +00:00
|
|
|
int delta = is_control ? 1 : (is_alt ? 32 : 8);
|
2007-04-29 17:35:43 +00:00
|
|
|
QPoint pos = cursorPos();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
switch(kc) {
|
|
|
|
case XK_Left:
|
|
|
|
case XK_KP_Left:
|
|
|
|
pos.rx() -= delta;
|
|
|
|
break;
|
|
|
|
case XK_Right:
|
|
|
|
case XK_KP_Right:
|
|
|
|
pos.rx() += delta;
|
|
|
|
break;
|
|
|
|
case XK_Up:
|
|
|
|
case XK_KP_Up:
|
|
|
|
pos.ry() -= delta;
|
|
|
|
break;
|
|
|
|
case XK_Down:
|
|
|
|
case XK_KP_Down:
|
|
|
|
pos.ry() += delta;
|
|
|
|
break;
|
|
|
|
case XK_F1:
|
|
|
|
if (!mouse_emulation_state)
|
|
|
|
mouse_emulation_window = getMouseEmulationWindow();
|
|
|
|
if ((mouse_emulation_state & Button1Mask) == 0)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuPress, Button1, mouse_emulation_state);
|
|
|
|
if (!is_shift)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button1, mouse_emulation_state);
|
|
|
|
break;
|
|
|
|
case XK_F2:
|
|
|
|
if (!mouse_emulation_state)
|
|
|
|
mouse_emulation_window = getMouseEmulationWindow();
|
|
|
|
if ((mouse_emulation_state & Button2Mask) == 0)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuPress, Button2, mouse_emulation_state);
|
|
|
|
if (!is_shift)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button2, mouse_emulation_state);
|
|
|
|
break;
|
|
|
|
case XK_F3:
|
|
|
|
if (!mouse_emulation_state)
|
|
|
|
mouse_emulation_window = getMouseEmulationWindow();
|
|
|
|
if ((mouse_emulation_state & Button3Mask) == 0)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuPress, Button3, mouse_emulation_state);
|
|
|
|
if (!is_shift)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button3, mouse_emulation_state);
|
|
|
|
break;
|
|
|
|
case XK_Return:
|
|
|
|
case XK_space:
|
|
|
|
case XK_KP_Enter:
|
|
|
|
case XK_KP_Space: {
|
|
|
|
if (!mouse_emulation_state) {
|
|
|
|
// Nothing was pressed, fake a LMB click
|
|
|
|
mouse_emulation_window = getMouseEmulationWindow();
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuPress, Button1, mouse_emulation_state);
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button1, mouse_emulation_state);
|
|
|
|
} else {
|
|
|
|
// Release all
|
|
|
|
if (mouse_emulation_state & Button1Mask)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button1, mouse_emulation_state);
|
|
|
|
if (mouse_emulation_state & Button2Mask)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button2, mouse_emulation_state);
|
|
|
|
if (mouse_emulation_state & Button3Mask)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuRelease, Button3, mouse_emulation_state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Fall through
|
|
|
|
case XK_Escape:
|
|
|
|
ungrabXKeyboard();
|
|
|
|
mouse_emulation = false;
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
QCursor::setPos(pos);
|
|
|
|
if (mouse_emulation_state)
|
|
|
|
mouse_emulation_state = sendFakedMouseEvent(pos, mouse_emulation_window,
|
|
|
|
EmuMove, 0, mouse_emulation_state);
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
return true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Delayed focus functions
|
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::delayFocus()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
requestFocus(delayfocus_client);
|
2007-04-29 17:35:43 +00:00
|
|
|
cancelDelayFocus();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::requestDelayFocus(Client* c)
|
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
delayfocus_client = c;
|
|
|
|
delete delayFocusTimer;
|
2011-01-30 14:34:42 +00:00
|
|
|
delayFocusTimer = new QTimer(this);
|
|
|
|
connect(delayFocusTimer, SIGNAL(timeout()), this, SLOT(delayFocus()));
|
|
|
|
delayFocusTimer->setSingleShot(true);
|
|
|
|
delayFocusTimer->start(options->delayFocusInterval);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::cancelDelayFocus()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
delete delayFocusTimer;
|
|
|
|
delayFocusTimer = 0;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
2007-04-29 17:35:43 +00:00
|
|
|
// Electric Borders
|
2008-12-18 13:50:57 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// Electric Border Window management. Electric borders allow a user to change
|
|
|
|
// the virtual desktop or activate another features by moving the mouse pointer
|
|
|
|
// to the borders or corners. Technically this is done with input only windows.
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::updateElectricBorders()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
electric_time_first = xTime();
|
|
|
|
electric_time_last = xTime();
|
2009-02-10 06:22:04 +00:00
|
|
|
electric_time_last_trigger = xTime();
|
2007-04-29 17:35:43 +00:00
|
|
|
electric_current_border = ElectricNone;
|
2008-11-15 03:04:44 +00:00
|
|
|
QRect r = Kephal::ScreenUtils::desktopGeometry();
|
2007-04-29 17:35:43 +00:00
|
|
|
electricTop = r.top();
|
|
|
|
electricBottom = r.bottom();
|
|
|
|
electricLeft = r.left();
|
|
|
|
electricRight = r.right();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int pos = 0; pos < ELECTRIC_COUNT; ++pos) {
|
|
|
|
if (electric_reserved[pos] == 0) {
|
|
|
|
if (electric_windows[pos] != None)
|
|
|
|
XDestroyWindow(display(), electric_windows[pos]);
|
2008-12-18 13:50:57 +00:00
|
|
|
electric_windows[pos] = None;
|
2007-04-29 17:35:43 +00:00
|
|
|
continue;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if (electric_windows[pos] != None)
|
2007-04-29 17:35:43 +00:00
|
|
|
continue;
|
|
|
|
XSetWindowAttributes attributes;
|
|
|
|
attributes.override_redirect = True;
|
|
|
|
attributes.event_mask = EnterWindowMask | LeaveWindowMask;
|
|
|
|
unsigned long valuemask = CWOverrideRedirect | CWEventMask;
|
2011-01-30 14:34:42 +00:00
|
|
|
int xywh[ELECTRIC_COUNT][4] = {
|
|
|
|
{ r.left() + 1, r.top(), r.width() - 2, 1 }, // Top
|
|
|
|
{ r.right(), r.top(), 1, 1 }, // Top-right
|
|
|
|
{ r.right(), r.top() + 1, 1, r.height() - 2 }, // Etc.
|
|
|
|
{ r.right(), r.bottom(), 1, 1 },
|
|
|
|
{ r.left() + 1, r.bottom(), r.width() - 2, 1 },
|
|
|
|
{ r.left(), r.bottom(), 1, 1 },
|
|
|
|
{ r.left(), r.top() + 1, 1, r.height() - 2 },
|
|
|
|
{ r.left(), r.top(), 1, 1 }
|
|
|
|
};
|
|
|
|
electric_windows[pos] = XCreateWindow(display(), rootWindow(),
|
|
|
|
xywh[pos][0], xywh[pos][1], xywh[pos][2], xywh[pos][3],
|
|
|
|
0, CopyFromParent, InputOnly, CopyFromParent, valuemask, &attributes);
|
|
|
|
XMapWindow(display(), electric_windows[pos]);
|
2008-12-18 13:50:57 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
// Set XdndAware on the windows, so that DND enter events are received (#86998)
|
|
|
|
Atom version = 4; // XDND version
|
2011-01-30 14:34:42 +00:00
|
|
|
XChangeProperty(display(), electric_windows[pos], atoms->xdnd_aware, XA_ATOM,
|
|
|
|
32, PropModeReplace, (unsigned char*)(&version), 1);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::destroyElectricBorders()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
for (int pos = 0; pos < ELECTRIC_COUNT; ++pos) {
|
|
|
|
if (electric_windows[pos] != None)
|
|
|
|
XDestroyWindow(display(), electric_windows[pos]);
|
2008-12-18 13:50:57 +00:00
|
|
|
electric_windows[pos] = None;
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::restoreElectricBorderSize(ElectricBorder border)
|
|
|
|
{
|
|
|
|
if (electric_windows[border] == None)
|
2009-10-06 15:52:08 +00:00
|
|
|
return;
|
|
|
|
QRect r = Kephal::ScreenUtils::desktopGeometry();
|
2011-01-30 14:34:42 +00:00
|
|
|
int xywh[ELECTRIC_COUNT][4] = {
|
|
|
|
{ r.left() + 1, r.top(), r.width() - 2, 1 }, // Top
|
|
|
|
{ r.right(), r.top(), 1, 1 }, // Top-right
|
|
|
|
{ r.right(), r.top() + 1, 1, r.height() - 2 }, // Etc.
|
|
|
|
{ r.right(), r.bottom(), 1, 1 },
|
|
|
|
{ r.left() + 1, r.bottom(), r.width() - 2, 1 },
|
|
|
|
{ r.left(), r.bottom(), 1, 1 },
|
|
|
|
{ r.left(), r.top() + 1, 1, r.height() - 2 },
|
|
|
|
{ r.left(), r.top(), 1, 1 }
|
|
|
|
};
|
|
|
|
XMoveResizeWindow(display(), electric_windows[border],
|
|
|
|
xywh[border][0], xywh[border][1], xywh[border][2], xywh[border][3]);
|
|
|
|
}
|
2009-10-06 15:52:08 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::reserveElectricBorderActions(bool reserve)
|
|
|
|
{
|
|
|
|
for (int pos = 0; pos < ELECTRIC_COUNT; ++pos)
|
|
|
|
if (options->electricBorderAction(static_cast<ElectricBorder>(pos))) {
|
|
|
|
if (reserve)
|
|
|
|
reserveElectricBorder(static_cast<ElectricBorder>(pos));
|
2009-02-28 08:33:16 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
unreserveElectricBorder(static_cast<ElectricBorder>(pos));
|
|
|
|
}
|
|
|
|
}
|
2009-02-28 08:33:16 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::reserveElectricBorderSwitching(bool reserve)
|
|
|
|
{
|
|
|
|
for (int pos = 0; pos < ELECTRIC_COUNT; ++pos)
|
|
|
|
if (reserve)
|
|
|
|
reserveElectricBorder(static_cast<ElectricBorder>(pos));
|
2007-06-15 13:06:17 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
unreserveElectricBorder(static_cast<ElectricBorder>(pos));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::reserveElectricBorder(ElectricBorder border)
|
|
|
|
{
|
|
|
|
if (border == ElectricNone)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (electric_reserved[border]++ == 0)
|
|
|
|
QTimer::singleShot(0, this, SLOT(updateElectricBorders()));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::unreserveElectricBorder(ElectricBorder border)
|
|
|
|
{
|
|
|
|
if (border == ElectricNone)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
assert(electric_reserved[border] > 0);
|
|
|
|
if (--electric_reserved[border] == 0)
|
|
|
|
QTimer::singleShot(0, this, SLOT(updateElectricBorders()));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
void Workspace::checkElectricBorder(const QPoint& pos, Time now)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if ((pos.x() != electricLeft) &&
|
|
|
|
(pos.x() != electricRight) &&
|
|
|
|
(pos.y() != electricTop) &&
|
|
|
|
(pos.y() != electricBottom))
|
|
|
|
return;
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
bool have_borders = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (int i = 0; i < ELECTRIC_COUNT; ++i)
|
|
|
|
if (electric_windows[i] != None)
|
2007-04-29 17:35:43 +00:00
|
|
|
have_borders = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!have_borders)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
Time treshold_set = options->electricBorderDelay(); // Set timeout
|
|
|
|
Time treshold_reset = 250; // Reset timeout
|
2009-02-28 08:33:16 +00:00
|
|
|
Time treshold_trigger = options->electricBorderCooldown(); // Minimum time between triggers
|
2007-04-29 17:35:43 +00:00
|
|
|
int distance_reset = 30; // Mouse should not move more than this many pixels
|
2009-09-08 20:01:16 +00:00
|
|
|
int pushback_pixels = options->electricBorderPushbackPixels();
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
ElectricBorder border;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (pos.x() == electricLeft && pos.y() == electricTop)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricTopLeft;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.x() == electricRight && pos.y() == electricTop)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricTopRight;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.x() == electricLeft && pos.y() == electricBottom)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricBottomLeft;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.x() == electricRight && pos.y() == electricBottom)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricBottomRight;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.x() == electricLeft)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricLeft;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.x() == electricRight)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricRight;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.y() == electricTop)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricTop;
|
2011-01-30 14:34:42 +00:00
|
|
|
else if (pos.y() == electricBottom)
|
2007-04-29 17:35:43 +00:00
|
|
|
border = ElectricBottom;
|
|
|
|
else
|
|
|
|
abort();
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (electric_windows[border] == None)
|
2007-06-15 10:37:34 +00:00
|
|
|
return;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (pushback_pixels == 0) {
|
2009-09-08 20:01:16 +00:00
|
|
|
// no pushback so we have to activate at once
|
|
|
|
electric_time_last = now;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
if ((electric_current_border == border) &&
|
|
|
|
(timestampDiff(electric_time_last, now) < treshold_reset) &&
|
|
|
|
(timestampDiff(electric_time_last_trigger, now) > treshold_trigger) &&
|
|
|
|
((pos - electric_push_point).manhattanLength() < distance_reset)) {
|
2007-04-29 17:35:43 +00:00
|
|
|
electric_time_last = now;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
if (timestampDiff(electric_time_first, now) > treshold_set) {
|
2007-04-29 17:35:43 +00:00
|
|
|
electric_current_border = ElectricNone;
|
2009-02-10 06:22:04 +00:00
|
|
|
electric_time_last_trigger = now;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (movingClient) {
|
2009-09-08 20:01:08 +00:00
|
|
|
// If moving a client or have force doing the desktop switch
|
2011-01-30 14:34:42 +00:00
|
|
|
if (options->electricBorders() != Options::ElectricDisabled)
|
|
|
|
electricBorderSwitchDesktop(border, pos);
|
2010-01-09 09:49:52 +00:00
|
|
|
return; // Don't reset cursor position
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
|
|
|
if (options->electricBorders() == Options::ElectricAlways &&
|
|
|
|
(border == ElectricTop || border == ElectricRight ||
|
|
|
|
border == ElectricBottom || border == ElectricLeft)) {
|
|
|
|
// If desktop switching is always enabled don't apply it to the corners if
|
2009-09-08 20:01:08 +00:00
|
|
|
// an effect is applied to it (We will check that later).
|
2011-01-30 14:34:42 +00:00
|
|
|
electricBorderSwitchDesktop(border, pos);
|
2009-09-08 20:01:08 +00:00
|
|
|
return; // Don't reset cursor position
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
switch(options->electricBorderAction(border)) {
|
|
|
|
case ElectricActionDashboard: { // Display Plasma dashboard
|
|
|
|
QDBusInterface plasmaApp("org.kde.plasma-desktop", "/App");
|
|
|
|
plasmaApp.call("toggleDashboard");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ElectricActionShowDesktop: {
|
|
|
|
setShowingDesktop(!showingDesktop());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ElectricActionLockScreen: { // Lock the screen
|
|
|
|
QDBusInterface screenSaver("org.kde.screensaver", "/ScreenSaver");
|
|
|
|
screenSaver.call("Lock");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ElectricActionPreventScreenLocking: {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ElectricActionNone: // Either desktop switching or an effect
|
|
|
|
default: {
|
|
|
|
if (effects && static_cast<EffectsHandlerImpl*>(effects)->borderActivated(border))
|
|
|
|
{} // Handled by effects
|
|
|
|
else {
|
|
|
|
electricBorderSwitchDesktop(border, pos);
|
|
|
|
return; // Don't reset cursor position
|
2009-09-08 20:01:08 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-02-10 06:22:04 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
2007-04-29 17:35:43 +00:00
|
|
|
electric_current_border = border;
|
|
|
|
electric_time_first = now;
|
|
|
|
electric_time_last = now;
|
|
|
|
electric_push_point = pos;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
// Reset the pointer to find out whether the user is really pushing
|
2007-04-29 17:35:43 +00:00
|
|
|
// (the direction back from which it came, starting from top clockwise)
|
2009-09-08 20:01:08 +00:00
|
|
|
const int xdiff[ELECTRIC_COUNT] = { 0,
|
|
|
|
-pushback_pixels,
|
|
|
|
-pushback_pixels,
|
|
|
|
-pushback_pixels,
|
|
|
|
0,
|
|
|
|
pushback_pixels,
|
|
|
|
pushback_pixels,
|
2011-01-30 14:34:42 +00:00
|
|
|
pushback_pixels
|
|
|
|
};
|
2009-09-08 20:01:08 +00:00
|
|
|
const int ydiff[ELECTRIC_COUNT] = { pushback_pixels,
|
|
|
|
pushback_pixels,
|
|
|
|
0,
|
|
|
|
-pushback_pixels,
|
|
|
|
-pushback_pixels,
|
|
|
|
-pushback_pixels,
|
|
|
|
0,
|
2011-01-30 14:34:42 +00:00
|
|
|
pushback_pixels
|
|
|
|
};
|
|
|
|
QCursor::setPos(pos.x() + xdiff[border], pos.y() + ydiff[border]);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::electricBorderSwitchDesktop(ElectricBorder border, const QPoint& _pos)
|
|
|
|
{
|
2007-06-15 13:06:17 +00:00
|
|
|
QPoint pos = _pos;
|
|
|
|
int desk = currentDesktop();
|
2007-06-21 15:07:14 +00:00
|
|
|
const int OFFSET = 2;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (border == ElectricLeft || border == ElectricTopLeft || border == ElectricBottomLeft) {
|
|
|
|
desk = desktopToLeft(desk, options->rollOverDesktops);
|
|
|
|
pos.setX(displayWidth() - 1 - OFFSET);
|
|
|
|
}
|
|
|
|
if (border == ElectricRight || border == ElectricTopRight || border == ElectricBottomRight) {
|
|
|
|
desk = desktopToRight(desk, options->rollOverDesktops);
|
|
|
|
pos.setX(OFFSET);
|
|
|
|
}
|
|
|
|
if (border == ElectricTop || border == ElectricTopLeft || border == ElectricTopRight) {
|
|
|
|
desk = desktopAbove(desk, options->rollOverDesktops);
|
|
|
|
pos.setY(displayHeight() - 1 - OFFSET);
|
|
|
|
}
|
|
|
|
if (border == ElectricBottom || border == ElectricBottomLeft || border == ElectricBottomRight) {
|
|
|
|
desk = desktopBelow(desk, options->rollOverDesktops);
|
|
|
|
pos.setY(OFFSET);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
int desk_before = currentDesktop();
|
|
|
|
setCurrentDesktop(desk);
|
|
|
|
if (currentDesktop() != desk_before)
|
|
|
|
QCursor::setPos(pos);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Called when the user entered an electric border with the mouse.
|
|
|
|
* It may switch to another virtual desktop.
|
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
bool Workspace::electricBorderEvent(XEvent* e)
|
|
|
|
{
|
|
|
|
if (e->type == EnterNotify) {
|
|
|
|
for (int i = 0; i < ELECTRIC_COUNT; ++i)
|
|
|
|
if (electric_windows[i] != None && e->xcrossing.window == electric_windows[i]) {
|
|
|
|
// The user entered an electric border
|
|
|
|
checkElectricBorder(QPoint(e->xcrossing.x_root, e->xcrossing.y_root), e->xcrossing.time);
|
2007-04-29 17:35:43 +00:00
|
|
|
return true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (e->type == ClientMessage) {
|
|
|
|
if (e->xclient.message_type == atoms->xdnd_position) {
|
|
|
|
for (int i = 0; i < ELECTRIC_COUNT; ++i)
|
|
|
|
if (electric_windows[i] != None && e->xclient.window == electric_windows[i]) {
|
2007-04-29 17:35:43 +00:00
|
|
|
updateXTime();
|
2011-01-30 14:34:42 +00:00
|
|
|
checkElectricBorder(QPoint(
|
|
|
|
e->xclient.data.l[2] >> 16, e->xclient.data.l[2] & 0xffff), xTime());
|
2007-04-29 17:35:43 +00:00
|
|
|
return true;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
return false;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2009-09-08 20:01:08 +00:00
|
|
|
void Workspace::showElectricBorderWindowOutline()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (!movingClient)
|
2009-09-08 20:01:08 +00:00
|
|
|
return;
|
|
|
|
// code copied from TabBox::updateOutline() in tabbox.cpp
|
|
|
|
QRect c = movingClient->electricBorderMaximizeGeometry();
|
|
|
|
// left/right parts are between top/bottom, they don't reach as far as the corners
|
2011-01-30 14:34:42 +00:00
|
|
|
XMoveResizeWindow(QX11Info::display(), outline_left, c.x(), c.y() + 5, 5, c.height() - 10);
|
|
|
|
XMoveResizeWindow(QX11Info::display(), outline_right, c.x() + c.width() - 5, c.y() + 5, 5, c.height() - 10);
|
|
|
|
XMoveResizeWindow(QX11Info::display(), outline_top, c.x(), c.y(), c.width(), 5);
|
|
|
|
XMoveResizeWindow(QX11Info::display(), outline_bottom, c.x(), c.y() + c.height() - 5, c.width(), 5);
|
|
|
|
{
|
|
|
|
QPixmap pix(5, c.height() - 10);
|
|
|
|
QPainter p(&pix);
|
|
|
|
p.setPen(Qt::white);
|
|
|
|
p.drawLine(0, 0, 0, pix.height() - 1);
|
|
|
|
p.drawLine(4, 0, 4, pix.height() - 1);
|
|
|
|
p.setPen(Qt::gray);
|
|
|
|
p.drawLine(1, 0, 1, pix.height() - 1);
|
|
|
|
p.drawLine(3, 0, 3, pix.height() - 1);
|
|
|
|
p.setPen(Qt::black);
|
|
|
|
p.drawLine(2, 0, 2, pix.height() - 1);
|
|
|
|
p.end();
|
|
|
|
XSetWindowBackgroundPixmap(QX11Info::display(), outline_left, pix.handle());
|
|
|
|
XSetWindowBackgroundPixmap(QX11Info::display(), outline_right, pix.handle());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
QPixmap pix(c.width(), 5);
|
|
|
|
QPainter p(&pix);
|
|
|
|
p.setPen(Qt::white);
|
|
|
|
p.drawLine(0, 0, pix.width() - 1 - 0, 0);
|
|
|
|
p.drawLine(4, 4, pix.width() - 1 - 4, 4);
|
|
|
|
p.drawLine(0, 0, 0, 4);
|
|
|
|
p.drawLine(pix.width() - 1 - 0, 0, pix.width() - 1 - 0, 4);
|
|
|
|
p.setPen(Qt::gray);
|
|
|
|
p.drawLine(1, 1, pix.width() - 1 - 1, 1);
|
|
|
|
p.drawLine(3, 3, pix.width() - 1 - 3, 3);
|
|
|
|
p.drawLine(1, 1, 1, 4);
|
|
|
|
p.drawLine(3, 3, 3, 4);
|
|
|
|
p.drawLine(pix.width() - 1 - 1, 1, pix.width() - 1 - 1, 4);
|
|
|
|
p.drawLine(pix.width() - 1 - 3, 3, pix.width() - 1 - 3, 4);
|
|
|
|
p.setPen(Qt::black);
|
|
|
|
p.drawLine(2, 2, pix.width() - 1 - 2, 2);
|
|
|
|
p.drawLine(2, 2, 2, 4);
|
|
|
|
p.drawLine(pix.width() - 1 - 2, 2, pix.width() - 1 - 2, 4);
|
|
|
|
p.end();
|
|
|
|
XSetWindowBackgroundPixmap(QX11Info::display(), outline_top, pix.handle());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
QPixmap pix(c.width(), 5);
|
|
|
|
QPainter p(&pix);
|
|
|
|
p.setPen(Qt::white);
|
|
|
|
p.drawLine(4, 0, pix.width() - 1 - 4, 0);
|
|
|
|
p.drawLine(0, 4, pix.width() - 1 - 0, 4);
|
|
|
|
p.drawLine(0, 4, 0, 0);
|
|
|
|
p.drawLine(pix.width() - 1 - 0, 4, pix.width() - 1 - 0, 0);
|
|
|
|
p.setPen(Qt::gray);
|
|
|
|
p.drawLine(3, 1, pix.width() - 1 - 3, 1);
|
|
|
|
p.drawLine(1, 3, pix.width() - 1 - 1, 3);
|
|
|
|
p.drawLine(3, 1, 3, 0);
|
|
|
|
p.drawLine(1, 3, 1, 0);
|
|
|
|
p.drawLine(pix.width() - 1 - 3, 1, pix.width() - 1 - 3, 0);
|
|
|
|
p.drawLine(pix.width() - 1 - 1, 3, pix.width() - 1 - 1, 0);
|
|
|
|
p.setPen(Qt::black);
|
|
|
|
p.drawLine(2, 2, pix.width() - 1 - 2, 2);
|
|
|
|
p.drawLine(2, 0, 2, 2);
|
|
|
|
p.drawLine(pix.width() - 1 - 2, 0, pix.width() - 1 - 2, 2);
|
|
|
|
p.end();
|
|
|
|
XSetWindowBackgroundPixmap(QX11Info::display(), outline_bottom, pix.handle());
|
|
|
|
}
|
|
|
|
XClearWindow(QX11Info::display(), outline_left);
|
|
|
|
XClearWindow(QX11Info::display(), outline_right);
|
|
|
|
XClearWindow(QX11Info::display(), outline_top);
|
|
|
|
XClearWindow(QX11Info::display(), outline_bottom);
|
|
|
|
XMapWindow(QX11Info::display(), outline_left);
|
|
|
|
XMapWindow(QX11Info::display(), outline_right);
|
|
|
|
XMapWindow(QX11Info::display(), outline_top);
|
|
|
|
XMapWindow(QX11Info::display(), outline_bottom);
|
|
|
|
}
|
2009-09-08 20:01:08 +00:00
|
|
|
|
|
|
|
void Workspace::hideElectricBorderWindowOutline()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
XUnmapWindow(QX11Info::display(), outline_left);
|
|
|
|
XUnmapWindow(QX11Info::display(), outline_right);
|
|
|
|
XUnmapWindow(QX11Info::display(), outline_top);
|
|
|
|
XUnmapWindow(QX11Info::display(), outline_bottom);
|
|
|
|
}
|
2009-09-08 20:01:08 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// Top menu
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::addTopMenu(Client* c)
|
|
|
|
{
|
|
|
|
assert(c->isTopMenu());
|
|
|
|
assert(!topmenus.contains(c));
|
|
|
|
topmenus.append(c);
|
|
|
|
if (managingTopMenus()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
int minsize = c->minSize().height();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (minsize > topMenuHeight()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
topmenu_height = minsize;
|
|
|
|
updateTopMenuGeometry();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
updateTopMenuGeometry(c);
|
|
|
|
updateCurrentTopMenu();
|
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
|
|
|
|
//kDebug( 1212 ) << "NEW TOPMENU:" << c;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::removeTopMenu(Client* c)
|
|
|
|
{
|
|
|
|
//if ( c->isTopMenu() )
|
2008-12-18 13:50:57 +00:00
|
|
|
// kDebug( 1212 ) << "REMOVE TOPMENU:" << c;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
assert(c->isTopMenu());
|
|
|
|
assert(topmenus.contains(c));
|
|
|
|
topmenus.removeAll(c);
|
2007-04-29 17:35:43 +00:00
|
|
|
updateCurrentTopMenu();
|
2008-12-18 13:50:57 +00:00
|
|
|
// TODO: Reduce topMenuHeight() if possible?
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::lostTopMenuSelection()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2008-12-18 13:50:57 +00:00
|
|
|
//kDebug( 1212 ) << "lost TopMenu selection";
|
|
|
|
|
|
|
|
// Make sure this signal is always set when not owning the selection
|
2011-01-30 14:34:42 +00:00
|
|
|
disconnect(topmenu_watcher, SIGNAL(lostOwner()), this, SLOT(lostTopMenuOwner()));
|
|
|
|
connect(topmenu_watcher, SIGNAL(lostOwner()), this, SLOT(lostTopMenuOwner()));
|
|
|
|
if (!managing_topmenus)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(topmenu_watcher, SIGNAL(lostOwner()), this, SLOT(lostTopMenuOwner()));
|
|
|
|
disconnect(topmenu_selection, SIGNAL(lostOwnership()), this, SLOT(lostTopMenuSelection()));
|
2007-04-29 17:35:43 +00:00
|
|
|
managing_topmenus = false;
|
|
|
|
delete topmenu_space;
|
|
|
|
topmenu_space = NULL;
|
|
|
|
updateClientArea();
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = topmenus.constBegin();
|
|
|
|
it != topmenus.constEnd();
|
|
|
|
++it)
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->checkWorkspacePosition();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::lostTopMenuOwner()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (!options->topMenuEnabled())
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2008-12-18 13:50:57 +00:00
|
|
|
//kDebug( 1212 ) << "TopMenu selection lost owner";
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!topmenu_selection->claim(false)) {
|
2008-12-18 13:50:57 +00:00
|
|
|
//kDebug( 1212 ) << "Failed to claim TopMenu selection";
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
//kDebug( 1212 ) << "Claimed TopMenu selection";
|
2007-04-29 17:35:43 +00:00
|
|
|
setupTopMenuHandling();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::setupTopMenuHandling()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (managing_topmenus)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
connect(topmenu_selection, SIGNAL(lostOwnership()), this, SLOT(lostTopMenuSelection()));
|
|
|
|
disconnect(topmenu_watcher, SIGNAL(lostOwner()), this, SLOT(lostTopMenuOwner()));
|
2007-04-29 17:35:43 +00:00
|
|
|
managing_topmenus = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
topmenu_space = new QWidget(NULL, Qt::X11BypassWindowManagerHint);
|
2008-12-18 13:50:57 +00:00
|
|
|
Window stack[2];
|
|
|
|
stack[0] = supportWindow->winId();
|
|
|
|
stack[1] = topmenu_space->winId();
|
2011-01-30 14:34:42 +00:00
|
|
|
XRestackWindows(display(), stack, 2);
|
2007-04-29 17:35:43 +00:00
|
|
|
updateTopMenuGeometry();
|
|
|
|
topmenu_space->show();
|
|
|
|
updateClientArea();
|
|
|
|
updateCurrentTopMenu();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
int Workspace::topMenuHeight() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (topmenu_height == 0) {
|
|
|
|
// Simply create a dummy menubar and use its preferred height as the menu height
|
2007-04-29 17:35:43 +00:00
|
|
|
KMenuBar tmpmenu;
|
2011-01-30 14:34:42 +00:00
|
|
|
tmpmenu.addAction("dummy");
|
2007-04-29 17:35:43 +00:00
|
|
|
topmenu_height = tmpmenu.sizeHint().height();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
return topmenu_height;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
KDecoration* Workspace::createDecoration(KDecorationBridge* bridge)
|
|
|
|
{
|
|
|
|
return mgr->createDecoration(bridge);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Returns a list of all colors (KDecorationDefines::ColorType) the current
|
|
|
|
* decoration supports
|
|
|
|
*/
|
|
|
|
QList<int> Workspace::decorationSupportedColors() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2008-03-21 22:17:10 +00:00
|
|
|
KDecorationFactory* factory = mgr->factory();
|
2008-12-18 13:50:57 +00:00
|
|
|
QList<int> ret;
|
2011-01-30 14:34:42 +00:00
|
|
|
for (Ability ab = ABILITYCOLOR_FIRST;
|
|
|
|
ab < ABILITYCOLOR_END;
|
|
|
|
ab = static_cast<Ability>(ab + 1))
|
|
|
|
if (factory->supports(ab))
|
2008-03-21 22:17:10 +00:00
|
|
|
ret << ab;
|
|
|
|
return ret;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-03-21 22:17:10 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
QString Workspace::desktopName(int desk) const
|
|
|
|
{
|
|
|
|
return QString::fromUtf8(rootInfo->desktopName(desk));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
bool Workspace::checkStartupNotification(Window w, KStartupInfoId& id, KStartupInfoData& data)
|
|
|
|
{
|
|
|
|
return startup->checkStartup(w, id, data) == KStartupInfo::Match;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Puts the focus on a dummy window
|
|
|
|
* Just using XSetInputFocus() with None would block keyboard input
|
2007-04-29 17:35:43 +00:00
|
|
|
*/
|
|
|
|
void Workspace::focusToNull()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
XSetInputFocus(display(), null_focus_window, RevertToPointerRoot, xTime());
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::helperDialog(const QString& message, const Client* c)
|
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
QStringList args;
|
|
|
|
QString type;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (message == "noborderaltf3") {
|
|
|
|
KAction* action = qobject_cast<KAction*>(keys->action("Window Operations Menu"));
|
|
|
|
assert(action != NULL);
|
|
|
|
QString shortcut = QString("%1 (%2)").arg(action->text())
|
|
|
|
.arg(action->globalShortcut().primary().toString(QKeySequence::NativeText));
|
2008-12-18 13:50:57 +00:00
|
|
|
args << "--msgbox" << i18n(
|
2011-01-30 14:34:42 +00:00
|
|
|
"You have selected to show a window without its border.\n"
|
|
|
|
"Without the border, you will not be able to enable the border "
|
|
|
|
"again using the mouse: use the window operations menu instead, "
|
|
|
|
"activated using the %1 keyboard shortcut.",
|
|
|
|
shortcut);
|
2007-04-29 17:35:43 +00:00
|
|
|
type = "altf3warning";
|
2011-01-30 14:34:42 +00:00
|
|
|
} else if (message == "fullscreenaltf3") {
|
|
|
|
KAction* action = qobject_cast<KAction*>(keys->action("Window Operations Menu"));
|
|
|
|
assert(action != NULL);
|
|
|
|
QString shortcut = QString("%1 (%2)").arg(action->text())
|
|
|
|
.arg(action->globalShortcut().primary().toString(QKeySequence::NativeText));
|
2008-12-18 13:50:57 +00:00
|
|
|
args << "--msgbox" << i18n(
|
2011-01-30 14:34:42 +00:00
|
|
|
"You have selected to show a window in fullscreen mode.\n"
|
|
|
|
"If the application itself does not have an option to turn the fullscreen "
|
|
|
|
"mode off you will not be able to disable it "
|
|
|
|
"again using the mouse: use the window operations menu instead, "
|
|
|
|
"activated using the %1 keyboard shortcut.",
|
|
|
|
shortcut);
|
2007-04-29 17:35:43 +00:00
|
|
|
type = "altf3warning";
|
2011-01-30 14:34:42 +00:00
|
|
|
} else
|
2008-09-17 11:44:51 +00:00
|
|
|
abort();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!type.isEmpty()) {
|
|
|
|
KConfig cfg("kwin_dialogsrc");
|
|
|
|
KConfigGroup cg(&cfg, "Notification Messages"); // Depends on KMessageBox
|
|
|
|
if (!cg.readEntry(type, true))
|
2008-12-18 13:50:57 +00:00
|
|
|
return;
|
|
|
|
args << "--dontagain" << "kwin_dialogsrc:" + type;
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c != NULL)
|
|
|
|
args << "--embed" << QString::number(c->window());
|
|
|
|
KProcess::startDetached("kdialog", args);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::setShowingDesktop(bool showing)
|
|
|
|
{
|
|
|
|
rootInfo->setShowingDesktop(showing);
|
2007-04-29 17:35:43 +00:00
|
|
|
showing_desktop = showing;
|
|
|
|
++block_showing_desktop;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (showing_desktop) {
|
2007-04-29 17:35:43 +00:00
|
|
|
showing_desktop_clients.clear();
|
|
|
|
++block_focus;
|
|
|
|
ClientList cls = stackingOrder();
|
2008-12-18 13:50:57 +00:00
|
|
|
// Find them first, then minimize, otherwise transients may get minimized with the window
|
2007-04-29 17:35:43 +00:00
|
|
|
// they're transient for
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = cls.constBegin();
|
|
|
|
it != cls.constEnd();
|
|
|
|
++it)
|
|
|
|
if ((*it)->isOnCurrentActivity() && (*it)->isOnCurrentDesktop() && (*it)->isShown(true) && !(*it)->isSpecialWindow())
|
|
|
|
showing_desktop_clients.prepend(*it); // Topmost first to reduce flicker
|
|
|
|
for (ClientList::ConstIterator it = showing_desktop_clients.constBegin();
|
|
|
|
it != showing_desktop_clients.constEnd();
|
|
|
|
++it)
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->minimize();
|
|
|
|
--block_focus;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (Client* desk = findDesktop(true, currentDesktop()))
|
|
|
|
requestFocus(desk);
|
|
|
|
} else {
|
|
|
|
for (ClientList::ConstIterator it = showing_desktop_clients.constBegin();
|
|
|
|
it != showing_desktop_clients.constEnd();
|
|
|
|
++it)
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->unminimize();
|
2011-01-30 14:34:42 +00:00
|
|
|
if (showing_desktop_clients.count() > 0)
|
|
|
|
requestFocus(showing_desktop_clients.first());
|
2007-04-29 17:35:43 +00:00
|
|
|
showing_desktop_clients.clear();
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
--block_showing_desktop;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Following Kicker's behavior:
|
|
|
|
* Changing a virtual desktop resets the state and shows the windows again.
|
|
|
|
* Unminimizing a window resets the state but keeps the windows hidden (except
|
|
|
|
* the one that was unminimized).
|
|
|
|
* A new window resets the state and shows the windows again, with the new window
|
|
|
|
* being active. Due to popular demand (#67406) by people who apparently
|
|
|
|
* don't see a difference between "show desktop" and "minimize all", this is not
|
|
|
|
* true if "showDesktopIsMinimizeAll" is set in kwinrc. In such case showing
|
|
|
|
* a new window resets the state but doesn't show windows.
|
|
|
|
*/
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::resetShowingDesktop(bool keep_hidden)
|
|
|
|
{
|
|
|
|
if (block_showing_desktop > 0)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
rootInfo->setShowingDesktop(false);
|
2007-04-29 17:35:43 +00:00
|
|
|
showing_desktop = false;
|
|
|
|
++block_showing_desktop;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!keep_hidden) {
|
|
|
|
for (ClientList::ConstIterator it = showing_desktop_clients.constBegin();
|
|
|
|
it != showing_desktop_clients.constEnd();
|
|
|
|
++it)
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->unminimize();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
showing_desktop_clients.clear();
|
|
|
|
--block_showing_desktop;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Activating/deactivating this feature works like this:
|
|
|
|
* When nothing is active, and the shortcut is pressed, global shortcuts are disabled
|
|
|
|
* (using global_shortcuts_disabled)
|
|
|
|
* When a window that has disabling forced is activated, global shortcuts are disabled.
|
|
|
|
* (using global_shortcuts_disabled_for_client)
|
|
|
|
* When a shortcut is pressed and global shortcuts are disabled (either by a shortcut
|
|
|
|
* or for a client), they are enabled again.
|
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::slotDisableGlobalShortcuts()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (global_shortcuts_disabled || global_shortcuts_disabled_for_client)
|
|
|
|
disableGlobalShortcuts(false);
|
2007-04-29 17:35:43 +00:00
|
|
|
else
|
2011-01-30 14:34:42 +00:00
|
|
|
disableGlobalShortcuts(true);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
static bool pending_dfc = false;
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::disableGlobalShortcutsForClient(bool disable)
|
|
|
|
{
|
|
|
|
if (global_shortcuts_disabled_for_client == disable)
|
2007-04-29 17:35:43 +00:00
|
|
|
return;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!global_shortcuts_disabled) {
|
|
|
|
if (disable)
|
2007-04-29 17:35:43 +00:00
|
|
|
pending_dfc = true;
|
2011-01-30 14:34:42 +00:00
|
|
|
KGlobalSettings::self()->emitChange(KGlobalSettings::BlockShortcuts, disable);
|
2008-12-18 13:50:57 +00:00
|
|
|
// KWin will get the kipc message too
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::disableGlobalShortcuts(bool disable)
|
|
|
|
{
|
|
|
|
KGlobalSettings::self()->emitChange(KGlobalSettings::BlockShortcuts, disable);
|
2008-12-18 13:50:57 +00:00
|
|
|
// KWin will get the kipc message too
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::slotBlockShortcuts(int data)
|
|
|
|
{
|
|
|
|
if (pending_dfc && data) {
|
2007-04-29 17:35:43 +00:00
|
|
|
global_shortcuts_disabled_for_client = true;
|
|
|
|
pending_dfc = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
2007-04-29 17:35:43 +00:00
|
|
|
global_shortcuts_disabled = data;
|
|
|
|
global_shortcuts_disabled_for_client = false;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2008-12-18 13:50:57 +00:00
|
|
|
// Update also Alt+LMB actions etc.
|
2011-01-30 14:34:42 +00:00
|
|
|
for (ClientList::ConstIterator it = clients.constBegin();
|
|
|
|
it != clients.constEnd();
|
|
|
|
++it)
|
2007-04-29 17:35:43 +00:00
|
|
|
(*it)->updateMouseGrab();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
// Optimized version of QCursor::pos() that tries to avoid X roundtrips
|
|
|
|
// by updating the value only when the X timestamp changes.
|
|
|
|
static QPoint last_cursor_pos;
|
|
|
|
static int last_buttons = 0;
|
|
|
|
static Time last_cursor_timestamp = CurrentTime;
|
2008-03-19 18:13:22 +00:00
|
|
|
static QTimer* last_cursor_timer;
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2007-08-24 09:54:21 +00:00
|
|
|
QPoint Workspace::cursorPos() const
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (last_cursor_timestamp == CurrentTime ||
|
|
|
|
last_cursor_timestamp != QX11Info::appTime()) {
|
2007-04-29 17:35:43 +00:00
|
|
|
last_cursor_timestamp = QX11Info::appTime();
|
|
|
|
Window root;
|
|
|
|
Window child;
|
|
|
|
int root_x, root_y, win_x, win_y;
|
|
|
|
uint state;
|
2011-01-30 14:34:42 +00:00
|
|
|
XQueryPointer(display(), rootWindow(), &root, &child,
|
|
|
|
&root_x, &root_y, &win_x, &win_y, &state);
|
|
|
|
last_cursor_pos = QPoint(root_x, root_y);
|
2007-04-29 17:35:43 +00:00
|
|
|
last_buttons = state;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (last_cursor_timer == NULL) {
|
|
|
|
Workspace* ws = const_cast<Workspace*>(this);
|
|
|
|
last_cursor_timer = new QTimer(ws);
|
|
|
|
last_cursor_timer->setSingleShot(true);
|
|
|
|
connect(last_cursor_timer, SIGNAL(timeout()), ws, SLOT(resetCursorPosTime()));
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
last_cursor_timer->start(0);
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
return last_cursor_pos;
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2008-12-18 13:50:57 +00:00
|
|
|
/**
|
|
|
|
* Because of QTimer's and the impossibility to get events for all mouse
|
|
|
|
* movements (at least I haven't figured out how) the position needs
|
|
|
|
* to be also refetched after each return to the event loop.
|
|
|
|
*/
|
2007-04-29 17:35:43 +00:00
|
|
|
void Workspace::resetCursorPosTime()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
last_cursor_timestamp = CurrentTime;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void Workspace::checkCursorPos()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
QPoint last = last_cursor_pos;
|
|
|
|
int lastb = last_buttons;
|
2008-12-18 13:50:57 +00:00
|
|
|
cursorPos(); // Update if needed
|
2011-01-30 14:34:42 +00:00
|
|
|
if (last != last_cursor_pos || lastb != last_buttons)
|
|
|
|
static_cast<EffectsHandlerImpl*>(effects)->mouseChanged(cursorPos(), last,
|
|
|
|
x11ToQtMouseButtons(last_buttons), x11ToQtMouseButtons(lastb),
|
|
|
|
x11ToQtKeyboardModifiers(last_buttons), x11ToQtKeyboardModifiers(lastb));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
int Workspace::indexOfClientGroup(ClientGroup* group)
|
|
|
|
{
|
|
|
|
return clientGroups.indexOf(group);
|
|
|
|
}
|
2009-11-15 03:24:04 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void Workspace::moveItemToClientGroup(ClientGroup* oldGroup, int oldIndex,
|
|
|
|
ClientGroup* group, int index)
|
|
|
|
{
|
|
|
|
Client* c = oldGroup->clients().at(oldIndex);
|
|
|
|
group->add(c, index, true);
|
|
|
|
}
|
2009-11-15 03:24:04 +00:00
|
|
|
|
2009-11-16 10:09:03 +00:00
|
|
|
// To accept "mainwindow#1" to "mainwindow#2"
|
2011-01-30 14:34:42 +00:00
|
|
|
static QByteArray truncatedWindowRole(QByteArray a)
|
|
|
|
{
|
2009-11-16 10:09:03 +00:00
|
|
|
int i = a.indexOf('#');
|
2011-01-30 14:34:42 +00:00
|
|
|
if (i == -1)
|
2009-11-16 10:09:03 +00:00
|
|
|
return a;
|
2011-01-30 14:34:42 +00:00
|
|
|
QByteArray b(a);
|
|
|
|
b.truncate(i);
|
2009-11-16 10:09:03 +00:00
|
|
|
return b;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-11-16 10:09:03 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
Client* Workspace::findSimilarClient(Client* c)
|
|
|
|
{
|
|
|
|
// Attempt to find a similar window to the input. If we find multiple possibilities that are in
|
|
|
|
// different groups then ignore all of them. This function is for automatic window grouping.
|
2009-11-16 10:09:03 +00:00
|
|
|
Client* found = NULL;
|
2009-11-16 13:31:02 +00:00
|
|
|
|
|
|
|
// See if the window has a group ID to match with
|
2011-01-30 14:34:42 +00:00
|
|
|
QString wGId = c->rules()->checkAutogroupById(QString());
|
|
|
|
if (!wGId.isEmpty()) {
|
|
|
|
foreach (Client * cl, clients) {
|
|
|
|
if (wGId == cl->rules()->checkAutogroupById(QString())) {
|
|
|
|
if (found && found->clientGroup() != cl->clientGroup()) { // We've found two, ignore both
|
2009-11-16 13:31:02 +00:00
|
|
|
found = NULL;
|
|
|
|
break; // Continue to the next test
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
found = cl;
|
2009-11-16 10:09:03 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
if (found)
|
|
|
|
return found;
|
|
|
|
}
|
2009-11-16 13:31:02 +00:00
|
|
|
|
2010-01-22 03:13:18 +00:00
|
|
|
// If this is a transient window don't take a guess
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c->isTransient())
|
2010-01-22 03:13:18 +00:00
|
|
|
return NULL;
|
|
|
|
|
2009-11-16 13:31:02 +00:00
|
|
|
// If we don't have an ID take a guess
|
2011-01-30 14:34:42 +00:00
|
|
|
if (c->rules()->checkAutogrouping(options->autogroupSimilarWindows)) {
|
|
|
|
QByteArray wRole = truncatedWindowRole(c->windowRole());
|
|
|
|
foreach (Client * cl, clients) {
|
|
|
|
QByteArray wRoleB = truncatedWindowRole(cl->windowRole());
|
|
|
|
if (c->resourceClass() == cl->resourceClass() && // Same resource class
|
|
|
|
wRole == wRoleB && // Same window role
|
|
|
|
cl->isNormalWindow()) { // Normal window TODO: Can modal windows be "normal"?
|
|
|
|
if (found && found->clientGroup() != cl->clientGroup()) // We've found two, ignore both
|
2009-11-16 13:31:02 +00:00
|
|
|
return NULL;
|
|
|
|
found = cl;
|
|
|
|
}
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-11-16 13:31:02 +00:00
|
|
|
|
2009-11-16 10:09:03 +00:00
|
|
|
return found;
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2009-11-16 10:09:03 +00:00
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
#include "workspace.moc"
|