2015-03-05 09:21:03 +00:00
|
|
|
/********************************************************************
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2015 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
|
|
|
|
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/>.
|
|
|
|
*********************************************************************/
|
|
|
|
#include "abstract_client.h"
|
2017-01-11 09:21:03 +00:00
|
|
|
|
|
|
|
#include "appmenu.h"
|
2015-12-03 16:27:00 +00:00
|
|
|
#include "decorations/decoratedclient.h"
|
2015-04-29 10:02:54 +00:00
|
|
|
#include "decorations/decorationpalette.h"
|
2016-05-23 08:01:31 +00:00
|
|
|
#include "decorations/decorationbridge.h"
|
2015-10-23 13:32:10 +00:00
|
|
|
#include "cursor.h"
|
2015-10-22 15:08:17 +00:00
|
|
|
#include "effects.h"
|
2015-03-13 11:54:11 +00:00
|
|
|
#include "focuschain.h"
|
2015-10-22 13:58:24 +00:00
|
|
|
#include "outline.h"
|
2015-07-09 14:08:01 +00:00
|
|
|
#include "screens.h"
|
2015-03-12 14:21:07 +00:00
|
|
|
#ifdef KWIN_BUILD_TABBOX
|
|
|
|
#include "tabbox.h"
|
|
|
|
#endif
|
2015-10-22 13:58:24 +00:00
|
|
|
#include "screenedge.h"
|
2015-07-09 14:08:01 +00:00
|
|
|
#include "tabgroup.h"
|
2016-02-18 10:30:52 +00:00
|
|
|
#include "useractions.h"
|
2015-03-12 15:08:19 +00:00
|
|
|
#include "workspace.h"
|
2015-03-05 09:21:03 +00:00
|
|
|
|
2015-07-09 07:10:33 +00:00
|
|
|
#include "wayland_server.h"
|
|
|
|
#include <KWayland/Server/plasmawindowmanagement_interface.h>
|
|
|
|
|
2015-12-03 12:24:44 +00:00
|
|
|
#include <KDecoration2/Decoration>
|
|
|
|
|
2016-10-27 13:59:08 +00:00
|
|
|
#include <KDesktopFile>
|
|
|
|
|
2015-12-03 15:40:27 +00:00
|
|
|
#include <QMouseEvent>
|
|
|
|
#include <QStyleHints>
|
|
|
|
|
2015-03-05 09:21:03 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2015-04-29 10:02:54 +00:00
|
|
|
QHash<QString, std::weak_ptr<Decoration::DecorationPalette>> AbstractClient::s_palettes;
|
|
|
|
std::shared_ptr<Decoration::DecorationPalette> AbstractClient::s_defaultPalette;
|
|
|
|
|
2015-03-05 09:21:03 +00:00
|
|
|
AbstractClient::AbstractClient()
|
|
|
|
: Toplevel()
|
2015-03-12 14:21:07 +00:00
|
|
|
#ifdef KWIN_BUILD_TABBOX
|
|
|
|
, m_tabBoxClient(QSharedPointer<TabBox::TabBoxClientImpl>(new TabBox::TabBoxClientImpl(this)))
|
|
|
|
#endif
|
2015-04-29 10:02:54 +00:00
|
|
|
, m_colorScheme(QStringLiteral("kdeglobals"))
|
2015-03-05 09:21:03 +00:00
|
|
|
{
|
2015-10-26 09:14:54 +00:00
|
|
|
connect(this, &AbstractClient::geometryShapeChanged, this, &AbstractClient::geometryChanged);
|
|
|
|
auto signalMaximizeChanged = static_cast<void (AbstractClient::*)(KWin::AbstractClient*, MaximizeMode)>(&AbstractClient::clientMaximizedStateChanged);
|
|
|
|
connect(this, signalMaximizeChanged, this, &AbstractClient::geometryChanged);
|
|
|
|
connect(this, &AbstractClient::clientStepUserMovedResized, this, &AbstractClient::geometryChanged);
|
|
|
|
connect(this, &AbstractClient::clientStartUserMovedResized, this, &AbstractClient::moveResizedChanged);
|
|
|
|
connect(this, &AbstractClient::clientFinishUserMovedResized, this, &AbstractClient::moveResizedChanged);
|
|
|
|
connect(this, &AbstractClient::clientStartUserMovedResized, this, &AbstractClient::removeCheckScreenConnection);
|
|
|
|
connect(this, &AbstractClient::clientFinishUserMovedResized, this, &AbstractClient::setupCheckScreenConnection);
|
2015-12-03 12:31:25 +00:00
|
|
|
|
|
|
|
connect(this, &AbstractClient::paletteChanged, this, &AbstractClient::triggerDecorationRepaint);
|
2016-05-23 08:01:31 +00:00
|
|
|
|
|
|
|
connect(Decoration::DecorationBridge::self(), &QObject::destroyed, this, &AbstractClient::destroyDecoration);
|
2016-12-18 09:39:04 +00:00
|
|
|
|
|
|
|
// replace on-screen-display on size changes
|
|
|
|
connect(this, &AbstractClient::geometryShapeChanged, this,
|
|
|
|
[this] (Toplevel *c, const QRect &old) {
|
|
|
|
Q_UNUSED(c)
|
2017-10-12 18:33:30 +00:00
|
|
|
if (isOnScreenDisplay() && !geometry().isEmpty() && old.size() != geometry().size() && !isInitialPositionSet()) {
|
2016-12-26 10:04:14 +00:00
|
|
|
GeometryUpdatesBlocker blocker(this);
|
2016-12-18 09:39:04 +00:00
|
|
|
QRect area = workspace()->clientArea(PlacementArea, Screens::self()->current(), desktop());
|
|
|
|
Placement::self()->place(this, area);
|
|
|
|
setGeometryRestore(geometry());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2017-01-11 09:21:03 +00:00
|
|
|
|
2017-03-17 16:46:13 +00:00
|
|
|
connect(this, &AbstractClient::paddingChanged, this, [this]() {
|
|
|
|
m_visibleRectBeforeGeometryUpdate = visibleRect();
|
|
|
|
});
|
|
|
|
|
2017-01-11 09:21:03 +00:00
|
|
|
connect(ApplicationMenu::self(), &ApplicationMenu::applicationMenuEnabledChanged, this, [this] {
|
|
|
|
emit hasApplicationMenuChanged(hasApplicationMenu());
|
|
|
|
});
|
2015-03-05 09:21:03 +00:00
|
|
|
}
|
|
|
|
|
2015-10-13 11:15:39 +00:00
|
|
|
AbstractClient::~AbstractClient()
|
|
|
|
{
|
|
|
|
assert(m_blockGeometryUpdates == 0);
|
2015-12-04 07:12:49 +00:00
|
|
|
Q_ASSERT(m_decoration.decoration == nullptr);
|
2015-10-13 11:15:39 +00:00
|
|
|
}
|
2015-03-05 09:21:03 +00:00
|
|
|
|
|
|
|
void AbstractClient::updateMouseGrab()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-11-05 09:10:17 +00:00
|
|
|
bool AbstractClient::belongToSameApplication(const AbstractClient *c1, const AbstractClient *c2, SameApplicationChecks checks)
|
2015-03-05 09:21:03 +00:00
|
|
|
{
|
2017-11-05 09:10:17 +00:00
|
|
|
return c1->belongsToSameApplication(c2, checks);
|
2015-03-05 09:21:03 +00:00
|
|
|
}
|
|
|
|
|
2015-03-05 12:35:54 +00:00
|
|
|
bool AbstractClient::isTransient() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-03-06 09:05:40 +00:00
|
|
|
TabGroup *AbstractClient::tabGroup() const
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AbstractClient::untab(const QRect &toGeometry, bool clientRemoved)
|
|
|
|
{
|
|
|
|
Q_UNUSED(toGeometry)
|
|
|
|
Q_UNUSED(clientRemoved)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-03-12 09:43:46 +00:00
|
|
|
bool AbstractClient::isCurrentTab() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-03-12 10:24:27 +00:00
|
|
|
xcb_timestamp_t AbstractClient::userTime() const
|
|
|
|
{
|
|
|
|
return XCB_TIME_CURRENT_TIME;
|
|
|
|
}
|
|
|
|
|
2015-03-12 14:35:36 +00:00
|
|
|
void AbstractClient::setSkipSwitcher(bool set)
|
|
|
|
{
|
|
|
|
set = rules()->checkSkipSwitcher(set);
|
|
|
|
if (set == skipSwitcher())
|
|
|
|
return;
|
|
|
|
m_skipSwitcher = set;
|
|
|
|
updateWindowRules(Rules::SkipSwitcher);
|
|
|
|
emit skipSwitcherChanged();
|
|
|
|
}
|
|
|
|
|
2015-06-06 19:17:23 +00:00
|
|
|
void AbstractClient::setSkipPager(bool b)
|
|
|
|
{
|
|
|
|
b = rules()->checkSkipPager(b);
|
|
|
|
if (b == skipPager())
|
|
|
|
return;
|
|
|
|
m_skipPager = b;
|
|
|
|
doSetSkipPager();
|
|
|
|
updateWindowRules(Rules::SkipPager);
|
|
|
|
emit skipPagerChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetSkipPager()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-06-06 20:08:12 +00:00
|
|
|
void AbstractClient::setSkipTaskbar(bool b)
|
|
|
|
{
|
|
|
|
int was_wants_tab_focus = wantsTabFocus();
|
|
|
|
if (b == skipTaskbar())
|
|
|
|
return;
|
|
|
|
m_skipTaskbar = b;
|
|
|
|
doSetSkipTaskbar();
|
|
|
|
updateWindowRules(Rules::SkipTaskbar);
|
|
|
|
if (was_wants_tab_focus != wantsTabFocus()) {
|
|
|
|
FocusChain::self()->update(this, isActive() ? FocusChain::MakeFirst : FocusChain::Update);
|
|
|
|
}
|
|
|
|
emit skipTaskbarChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setOriginalSkipTaskbar(bool b)
|
|
|
|
{
|
|
|
|
m_originalSkipTaskbar = rules()->checkSkipTaskbar(b);
|
|
|
|
setSkipTaskbar(m_originalSkipTaskbar);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetSkipTaskbar()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-03-12 14:44:39 +00:00
|
|
|
void AbstractClient::setIcon(const QIcon &icon)
|
|
|
|
{
|
|
|
|
m_icon = icon;
|
|
|
|
emit iconChanged();
|
|
|
|
}
|
|
|
|
|
2015-03-12 15:08:19 +00:00
|
|
|
void AbstractClient::setActive(bool act)
|
|
|
|
{
|
|
|
|
if (m_active == act) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_active = act;
|
|
|
|
const int ruledOpacity = m_active
|
|
|
|
? rules()->checkOpacityActive(qRound(opacity() * 100.0))
|
|
|
|
: rules()->checkOpacityInactive(qRound(opacity() * 100.0));
|
|
|
|
setOpacity(ruledOpacity / 100.0);
|
|
|
|
workspace()->setActiveClient(act ? this : NULL);
|
|
|
|
|
|
|
|
if (!m_active)
|
|
|
|
cancelAutoRaise();
|
|
|
|
|
|
|
|
if (!m_active && shadeMode() == ShadeActivated)
|
|
|
|
setShade(ShadeNormal);
|
|
|
|
|
2015-10-01 15:21:56 +00:00
|
|
|
StackingUpdatesBlocker blocker(workspace());
|
|
|
|
workspace()->updateClientLayer(this); // active windows may get different layer
|
|
|
|
auto mainclients = mainClients();
|
|
|
|
for (auto it = mainclients.constBegin();
|
|
|
|
it != mainclients.constEnd();
|
|
|
|
++it)
|
|
|
|
if ((*it)->isFullScreen()) // fullscreens go high even if their transient is active
|
|
|
|
workspace()->updateClientLayer(*it);
|
|
|
|
|
2015-03-12 15:08:19 +00:00
|
|
|
doSetActive();
|
|
|
|
emit activeChanged();
|
|
|
|
updateMouseGrab();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetActive()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-09-17 09:06:59 +00:00
|
|
|
Layer AbstractClient::layer() const
|
|
|
|
{
|
|
|
|
if (m_layer == UnknownLayer)
|
|
|
|
const_cast< AbstractClient* >(this)->m_layer = belongsToLayer();
|
|
|
|
return m_layer;
|
|
|
|
}
|
|
|
|
|
2015-03-13 08:26:09 +00:00
|
|
|
void AbstractClient::updateLayer()
|
2015-09-14 10:04:23 +00:00
|
|
|
{
|
2015-09-17 09:06:59 +00:00
|
|
|
if (layer() == belongsToLayer())
|
|
|
|
return;
|
2015-09-14 10:04:23 +00:00
|
|
|
StackingUpdatesBlocker blocker(workspace());
|
|
|
|
invalidateLayer(); // invalidate, will be updated when doing restacking
|
|
|
|
for (auto it = transients().constBegin(),
|
|
|
|
end = transients().constEnd(); it != end; ++it)
|
|
|
|
(*it)->updateLayer();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::invalidateLayer()
|
2015-03-13 08:26:09 +00:00
|
|
|
{
|
2015-09-17 09:06:59 +00:00
|
|
|
m_layer = UnknownLayer;
|
|
|
|
}
|
|
|
|
|
|
|
|
Layer AbstractClient::belongsToLayer() const
|
|
|
|
{
|
|
|
|
// NOTICE while showingDesktop, desktops move to the AboveLayer
|
|
|
|
// (interchangeable w/ eg. yakuake etc. which will at first remain visible)
|
|
|
|
// and the docks move into the NotificationLayer (which is between Above- and
|
|
|
|
// ActiveLayer, so that active fullscreen windows will still cover everything)
|
|
|
|
// Since the desktop is also activated, nothing should be in the ActiveLayer, though
|
|
|
|
if (isDesktop())
|
|
|
|
return workspace()->showingDesktop() ? AboveLayer : DesktopLayer;
|
|
|
|
if (isSplash()) // no damn annoying splashscreens
|
|
|
|
return NormalLayer; // getting in the way of everything else
|
|
|
|
if (isDock()) {
|
|
|
|
if (workspace()->showingDesktop())
|
|
|
|
return NotificationLayer;
|
|
|
|
return layerForDock();
|
|
|
|
}
|
|
|
|
if (isOnScreenDisplay())
|
|
|
|
return OnScreenDisplayLayer;
|
|
|
|
if (isNotification())
|
|
|
|
return NotificationLayer;
|
|
|
|
if (workspace()->showingDesktop() && belongsToDesktop()) {
|
|
|
|
return AboveLayer;
|
|
|
|
}
|
|
|
|
if (keepBelow())
|
|
|
|
return BelowLayer;
|
|
|
|
if (isActiveFullScreen())
|
|
|
|
return ActiveLayer;
|
|
|
|
if (keepAbove())
|
|
|
|
return AboveLayer;
|
|
|
|
|
|
|
|
return NormalLayer;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AbstractClient::belongsToDesktop() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Layer AbstractClient::layerForDock() const
|
|
|
|
{
|
|
|
|
// slight hack for the 'allow window to cover panel' Kicker setting
|
|
|
|
// don't move keepbelow docks below normal window, but only to the same
|
|
|
|
// layer, so that both may be raised to cover the other
|
|
|
|
if (keepBelow())
|
|
|
|
return NormalLayer;
|
|
|
|
if (keepAbove()) // slight hack for the autohiding panels
|
|
|
|
return AboveLayer;
|
|
|
|
return DockLayer;
|
2015-03-13 08:26:09 +00:00
|
|
|
}
|
|
|
|
|
2015-03-13 08:36:43 +00:00
|
|
|
void AbstractClient::setKeepAbove(bool b)
|
|
|
|
{
|
|
|
|
b = rules()->checkKeepAbove(b);
|
|
|
|
if (b && !rules()->checkKeepBelow(false))
|
|
|
|
setKeepBelow(false);
|
|
|
|
if (b == keepAbove()) {
|
|
|
|
// force hint change if different
|
|
|
|
if (info && bool(info->state() & NET::KeepAbove) != keepAbove())
|
|
|
|
info->setState(keepAbove() ? NET::KeepAbove : NET::States(0), NET::KeepAbove);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_keepAbove = b;
|
|
|
|
if (info) {
|
|
|
|
info->setState(keepAbove() ? NET::KeepAbove : NET::States(0), NET::KeepAbove);
|
|
|
|
}
|
|
|
|
workspace()->updateClientLayer(this);
|
|
|
|
updateWindowRules(Rules::Above);
|
|
|
|
|
|
|
|
doSetKeepAbove();
|
|
|
|
emit keepAboveChanged(m_keepAbove);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetKeepAbove()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setKeepBelow(bool b)
|
|
|
|
{
|
|
|
|
b = rules()->checkKeepBelow(b);
|
|
|
|
if (b && !rules()->checkKeepAbove(false))
|
|
|
|
setKeepAbove(false);
|
|
|
|
if (b == keepBelow()) {
|
|
|
|
// force hint change if different
|
|
|
|
if (info && bool(info->state() & NET::KeepBelow) != keepBelow())
|
|
|
|
info->setState(keepBelow() ? NET::KeepBelow : NET::States(0), NET::KeepBelow);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_keepBelow = b;
|
|
|
|
if (info) {
|
|
|
|
info->setState(keepBelow() ? NET::KeepBelow : NET::States(0), NET::KeepBelow);
|
|
|
|
}
|
|
|
|
workspace()->updateClientLayer(this);
|
|
|
|
updateWindowRules(Rules::Below);
|
|
|
|
|
|
|
|
doSetKeepBelow();
|
|
|
|
emit keepBelowChanged(m_keepBelow);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetKeepBelow()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-03-13 09:26:10 +00:00
|
|
|
void AbstractClient::startAutoRaise()
|
|
|
|
{
|
|
|
|
delete m_autoRaiseTimer;
|
|
|
|
m_autoRaiseTimer = new QTimer(this);
|
|
|
|
connect(m_autoRaiseTimer, &QTimer::timeout, this, &AbstractClient::autoRaise);
|
|
|
|
m_autoRaiseTimer->setSingleShot(true);
|
|
|
|
m_autoRaiseTimer->start(options->autoRaiseInterval());
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::cancelAutoRaise()
|
|
|
|
{
|
|
|
|
delete m_autoRaiseTimer;
|
|
|
|
m_autoRaiseTimer = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::autoRaise()
|
|
|
|
{
|
|
|
|
workspace()->raiseClient(this);
|
|
|
|
cancelAutoRaise();
|
|
|
|
}
|
|
|
|
|
2015-03-13 09:31:35 +00:00
|
|
|
bool AbstractClient::wantsTabFocus() const
|
|
|
|
{
|
|
|
|
return (isNormalWindow() || isDialog()) && wantsInput();
|
|
|
|
}
|
|
|
|
|
2015-03-13 09:33:31 +00:00
|
|
|
bool AbstractClient::isSpecialWindow() const
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
return isDesktop() || isDock() || isSplash() || isToolbar() || isNotification() || isOnScreenDisplay();
|
|
|
|
}
|
|
|
|
|
2015-03-13 10:19:46 +00:00
|
|
|
void AbstractClient::demandAttention(bool set)
|
|
|
|
{
|
|
|
|
if (isActive())
|
|
|
|
set = false;
|
|
|
|
if (m_demandsAttention == set)
|
|
|
|
return;
|
|
|
|
m_demandsAttention = set;
|
|
|
|
if (info) {
|
|
|
|
info->setState(set ? NET::DemandsAttention : NET::States(0), NET::DemandsAttention);
|
|
|
|
}
|
|
|
|
workspace()->clientAttentionChanged(this, set);
|
|
|
|
emit demandsAttentionChanged();
|
|
|
|
}
|
|
|
|
|
2015-03-13 11:54:11 +00:00
|
|
|
void AbstractClient::setDesktop(int desktop)
|
|
|
|
{
|
|
|
|
const int numberOfDesktops = VirtualDesktopManager::self()->count();
|
|
|
|
if (desktop != NET::OnAllDesktops) // Do range check
|
|
|
|
desktop = qMax(1, qMin(numberOfDesktops, desktop));
|
|
|
|
desktop = qMin(numberOfDesktops, rules()->checkDesktop(desktop));
|
|
|
|
if (m_desktop == desktop)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int was_desk = m_desktop;
|
|
|
|
const bool wasOnCurrentDesktop = isOnCurrentDesktop();
|
|
|
|
m_desktop = desktop;
|
|
|
|
|
2015-09-14 09:40:48 +00:00
|
|
|
if (info) {
|
|
|
|
info->setDesktop(desktop);
|
|
|
|
}
|
|
|
|
if ((was_desk == NET::OnAllDesktops) != (desktop == NET::OnAllDesktops)) {
|
|
|
|
// onAllDesktops changed
|
|
|
|
workspace()->updateOnAllDesktopsOfTransients(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto transients_stacking_order = workspace()->ensureStackingOrder(transients());
|
|
|
|
for (auto it = transients_stacking_order.constBegin();
|
|
|
|
it != transients_stacking_order.constEnd();
|
|
|
|
++it)
|
|
|
|
(*it)->setDesktop(desktop);
|
|
|
|
|
|
|
|
if (isModal()) // if a modal dialog is moved, move the mainwindow with it as otherwise
|
|
|
|
// the (just moved) modal dialog will confusingly return to the mainwindow with
|
|
|
|
// the next desktop change
|
|
|
|
{
|
|
|
|
foreach (AbstractClient * c2, mainClients())
|
|
|
|
c2->setDesktop(desktop);
|
|
|
|
}
|
|
|
|
|
2015-03-13 11:54:11 +00:00
|
|
|
doSetDesktop(desktop, was_desk);
|
|
|
|
|
|
|
|
FocusChain::self()->update(this, FocusChain::MakeFirst);
|
|
|
|
updateWindowRules(Rules::Desktop);
|
|
|
|
|
|
|
|
emit desktopChanged();
|
|
|
|
if (wasOnCurrentDesktop != isOnCurrentDesktop())
|
|
|
|
emit desktopPresenceChanged(this, was_desk);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetDesktop(int desktop, int was_desk)
|
|
|
|
{
|
|
|
|
Q_UNUSED(desktop)
|
|
|
|
Q_UNUSED(was_desk)
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setOnAllDesktops(bool b)
|
|
|
|
{
|
|
|
|
if ((b && isOnAllDesktops()) ||
|
|
|
|
(!b && !isOnAllDesktops()))
|
|
|
|
return;
|
|
|
|
if (b)
|
|
|
|
setDesktop(NET::OnAllDesktops);
|
|
|
|
else
|
|
|
|
setDesktop(VirtualDesktopManager::self()->current());
|
|
|
|
}
|
|
|
|
|
2015-03-13 12:39:53 +00:00
|
|
|
bool AbstractClient::isShadeable() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setShade(bool set)
|
|
|
|
{
|
|
|
|
set ? setShade(ShadeNormal) : setShade(ShadeNone);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setShade(ShadeMode mode)
|
|
|
|
{
|
|
|
|
Q_UNUSED(mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
ShadeMode AbstractClient::shadeMode() const
|
|
|
|
{
|
|
|
|
return ShadeNone;
|
|
|
|
}
|
|
|
|
|
2015-03-13 12:54:30 +00:00
|
|
|
AbstractClient::Position AbstractClient::titlebarPosition() const
|
|
|
|
{
|
|
|
|
// TODO: still needed, remove?
|
|
|
|
return PositionTop;
|
|
|
|
}
|
|
|
|
|
2016-05-12 08:28:40 +00:00
|
|
|
bool AbstractClient::titlebarPositionUnderMouse() const
|
|
|
|
{
|
|
|
|
if (!isDecorated()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const auto sectionUnderMouse = decoration()->sectionUnderMouse();
|
|
|
|
if (sectionUnderMouse == Qt::TitleBarArea) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// check other sections based on titlebarPosition
|
|
|
|
switch (titlebarPosition()) {
|
|
|
|
case AbstractClient::PositionTop:
|
|
|
|
return (sectionUnderMouse == Qt::TopLeftSection || sectionUnderMouse == Qt::TopSection || sectionUnderMouse == Qt::TopRightSection);
|
|
|
|
case AbstractClient::PositionLeft:
|
|
|
|
return (sectionUnderMouse == Qt::TopLeftSection || sectionUnderMouse == Qt::LeftSection || sectionUnderMouse == Qt::BottomLeftSection);
|
|
|
|
case AbstractClient::PositionRight:
|
|
|
|
return (sectionUnderMouse == Qt::BottomRightSection || sectionUnderMouse == Qt::RightSection || sectionUnderMouse == Qt::TopRightSection);
|
|
|
|
case AbstractClient::PositionBottom:
|
|
|
|
return (sectionUnderMouse == Qt::BottomLeftSection || sectionUnderMouse == Qt::BottomSection || sectionUnderMouse == Qt::BottomRightSection);
|
|
|
|
default:
|
|
|
|
// nothing
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-13 13:45:21 +00:00
|
|
|
void AbstractClient::setMinimized(bool set)
|
|
|
|
{
|
|
|
|
set ? minimize() : unminimize();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::minimize(bool avoid_animation)
|
|
|
|
{
|
|
|
|
if (!isMinimizable() || isMinimized())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (isShade() && info) // NETWM restriction - KWindowInfo::isMinimized() == Hidden && !Shaded
|
|
|
|
info->setState(0, NET::Shaded);
|
|
|
|
|
|
|
|
m_minimized = true;
|
|
|
|
|
|
|
|
doMinimize();
|
|
|
|
|
|
|
|
updateWindowRules(Rules::Minimize);
|
|
|
|
FocusChain::self()->update(this, FocusChain::MakeFirstMinimized);
|
|
|
|
// TODO: merge signal with s_minimized
|
|
|
|
emit clientMinimized(this, !avoid_animation);
|
|
|
|
emit minimizedChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::unminimize(bool avoid_animation)
|
|
|
|
{
|
|
|
|
if (!isMinimized())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (rules()->checkMinimize(false)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isShade() && info) // NETWM restriction - KWindowInfo::isMinimized() == Hidden && !Shaded
|
|
|
|
info->setState(NET::Shaded, NET::Shaded);
|
|
|
|
|
|
|
|
m_minimized = false;
|
|
|
|
|
|
|
|
doMinimize();
|
|
|
|
|
|
|
|
updateWindowRules(Rules::Minimize);
|
|
|
|
emit clientUnminimized(this, !avoid_animation);
|
|
|
|
emit minimizedChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doMinimize()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-04-29 10:02:54 +00:00
|
|
|
QPalette AbstractClient::palette() const
|
|
|
|
{
|
|
|
|
if (!m_palette) {
|
|
|
|
return QPalette();
|
|
|
|
}
|
|
|
|
return m_palette->palette();
|
|
|
|
}
|
|
|
|
|
|
|
|
const Decoration::DecorationPalette *AbstractClient::decorationPalette() const
|
|
|
|
{
|
|
|
|
return m_palette.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::updateColorScheme(QString path)
|
|
|
|
{
|
|
|
|
if (path.isEmpty()) {
|
|
|
|
path = QStringLiteral("kdeglobals");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_palette || m_colorScheme != path) {
|
|
|
|
m_colorScheme = path;
|
|
|
|
|
|
|
|
if (m_palette) {
|
|
|
|
disconnect(m_palette.get(), &Decoration::DecorationPalette::changed, this, &AbstractClient::handlePaletteChange);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto it = s_palettes.find(m_colorScheme);
|
|
|
|
|
|
|
|
if (it == s_palettes.end() || it->expired()) {
|
|
|
|
m_palette = std::make_shared<Decoration::DecorationPalette>(m_colorScheme);
|
|
|
|
if (m_palette->isValid()) {
|
|
|
|
s_palettes[m_colorScheme] = m_palette;
|
|
|
|
} else {
|
|
|
|
if (!s_defaultPalette) {
|
|
|
|
s_defaultPalette = std::make_shared<Decoration::DecorationPalette>(QStringLiteral("kdeglobals"));
|
|
|
|
s_palettes[QStringLiteral("kdeglobals")] = s_defaultPalette;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_palette = s_defaultPalette;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_colorScheme == QStringLiteral("kdeglobals")) {
|
|
|
|
s_defaultPalette = m_palette;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_palette = it->lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(m_palette.get(), &Decoration::DecorationPalette::changed, this, &AbstractClient::handlePaletteChange);
|
|
|
|
|
|
|
|
emit paletteChanged(palette());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::handlePaletteChange()
|
|
|
|
{
|
|
|
|
emit paletteChanged(palette());
|
|
|
|
}
|
|
|
|
|
2015-05-27 09:51:45 +00:00
|
|
|
void AbstractClient::keepInArea(QRect area, bool partial)
|
|
|
|
{
|
|
|
|
if (partial) {
|
|
|
|
// increase the area so that can have only 100 pixels in the area
|
|
|
|
area.setLeft(qMin(area.left() - width() + 100, area.left()));
|
|
|
|
area.setTop(qMin(area.top() - height() + 100, area.top()));
|
|
|
|
area.setRight(qMax(area.right() + width() - 100, area.right()));
|
|
|
|
area.setBottom(qMax(area.bottom() + height() - 100, area.bottom()));
|
|
|
|
}
|
|
|
|
if (!partial) {
|
|
|
|
// resize to fit into area
|
|
|
|
if (area.width() < width() || area.height() < height())
|
|
|
|
resizeWithChecks(qMin(area.width(), width()), qMin(area.height(), height()));
|
|
|
|
}
|
|
|
|
int tx = x(), ty = y();
|
|
|
|
if (geometry().right() > area.right() && width() <= area.width())
|
|
|
|
tx = area.right() - width() + 1;
|
|
|
|
if (geometry().bottom() > area.bottom() && height() <= area.height())
|
|
|
|
ty = area.bottom() - height() + 1;
|
|
|
|
if (!area.contains(geometry().topLeft())) {
|
|
|
|
if (tx < area.x())
|
|
|
|
tx = area.x();
|
|
|
|
if (ty < area.y())
|
|
|
|
ty = area.y();
|
|
|
|
}
|
|
|
|
if (tx != x() || ty != y())
|
|
|
|
move(tx, ty);
|
|
|
|
}
|
|
|
|
|
2015-05-27 11:21:56 +00:00
|
|
|
QSize AbstractClient::maxSize() const
|
|
|
|
{
|
|
|
|
return rules()->checkMaxSize(QSize(INT_MAX, INT_MAX));
|
|
|
|
}
|
|
|
|
|
|
|
|
QSize AbstractClient::minSize() const
|
|
|
|
{
|
|
|
|
return rules()->checkMinSize(QSize(0, 0));
|
|
|
|
}
|
|
|
|
|
2015-06-05 03:06:20 +00:00
|
|
|
void AbstractClient::updateMoveResize(const QPointF ¤tGlobalCursor)
|
|
|
|
{
|
2015-10-23 11:36:53 +00:00
|
|
|
handleMoveResize(pos(), currentGlobalCursor.toPoint());
|
2015-06-05 03:06:20 +00:00
|
|
|
}
|
|
|
|
|
2015-06-19 22:14:15 +00:00
|
|
|
bool AbstractClient::hasStrut() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-09 07:10:33 +00:00
|
|
|
void AbstractClient::setupWindowManagementInterface()
|
|
|
|
{
|
|
|
|
if (m_windowManagementInterface) {
|
|
|
|
// already setup
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!waylandServer() || !surface()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!waylandServer()->windowManagement()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
using namespace KWayland::Server;
|
2016-05-12 05:56:46 +00:00
|
|
|
auto w = waylandServer()->windowManagement()->createWindow(waylandServer()->windowManagement());
|
2015-07-09 07:10:33 +00:00
|
|
|
w->setTitle(caption());
|
|
|
|
w->setVirtualDesktop(isOnAllDesktops() ? 0 : desktop() - 1);
|
|
|
|
w->setActive(isActive());
|
|
|
|
w->setFullscreen(isFullScreen());
|
|
|
|
w->setKeepAbove(keepAbove());
|
|
|
|
w->setKeepBelow(keepBelow());
|
|
|
|
w->setMaximized(maximizeMode() == KWin::MaximizeFull);
|
|
|
|
w->setMinimized(isMinimized());
|
|
|
|
w->setOnAllDesktops(isOnAllDesktops());
|
|
|
|
w->setDemandsAttention(isDemandingAttention());
|
|
|
|
w->setCloseable(isCloseable());
|
|
|
|
w->setMaximizeable(isMaximizable());
|
|
|
|
w->setMinimizeable(isMinimizable());
|
|
|
|
w->setFullscreenable(isFullScreenable());
|
2016-10-13 11:57:34 +00:00
|
|
|
w->setIcon(icon());
|
2016-10-27 13:59:08 +00:00
|
|
|
auto updateAppId = [this, w] {
|
2017-05-11 16:05:36 +00:00
|
|
|
w->setAppId(QString::fromUtf8(m_desktopFileName.isEmpty() ? resourceClass() : m_desktopFileName));
|
2016-10-27 13:59:08 +00:00
|
|
|
};
|
|
|
|
updateAppId();
|
2015-09-29 18:25:04 +00:00
|
|
|
w->setSkipTaskbar(skipTaskbar());
|
2017-05-13 15:02:30 +00:00
|
|
|
w->setPid(pid());
|
2016-04-19 09:43:04 +00:00
|
|
|
w->setShadeable(isShadeable());
|
2016-04-15 11:41:10 +00:00
|
|
|
w->setShaded(isShade());
|
2016-04-17 14:43:45 +00:00
|
|
|
w->setResizable(isResizable());
|
|
|
|
w->setMovable(isMovable());
|
2016-04-19 16:34:59 +00:00
|
|
|
w->setVirtualDesktopChangeable(true); // FIXME Matches Client::actionSupported(), but both should be implemented.
|
2016-06-06 13:19:28 +00:00
|
|
|
w->setParentWindow(transientFor() ? transientFor()->windowManagementInterface() : nullptr);
|
2016-07-18 06:22:33 +00:00
|
|
|
w->setGeometry(geom);
|
2015-09-29 18:25:04 +00:00
|
|
|
connect(this, &AbstractClient::skipTaskbarChanged, w,
|
|
|
|
[w, this] {
|
|
|
|
w->setSkipTaskbar(skipTaskbar());
|
|
|
|
}
|
|
|
|
);
|
2015-07-09 07:10:33 +00:00
|
|
|
connect(this, &AbstractClient::captionChanged, w, [w, this] { w->setTitle(caption()); });
|
|
|
|
connect(this, &AbstractClient::desktopChanged, w,
|
|
|
|
[w, this] {
|
|
|
|
if (isOnAllDesktops()) {
|
|
|
|
w->setOnAllDesktops(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
w->setVirtualDesktop(desktop() - 1);
|
|
|
|
w->setOnAllDesktops(false);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(this, &AbstractClient::activeChanged, w, [w, this] { w->setActive(isActive()); });
|
|
|
|
connect(this, &AbstractClient::fullScreenChanged, w, [w, this] { w->setFullscreen(isFullScreen()); });
|
|
|
|
connect(this, &AbstractClient::keepAboveChanged, w, &PlasmaWindowInterface::setKeepAbove);
|
|
|
|
connect(this, &AbstractClient::keepBelowChanged, w, &PlasmaWindowInterface::setKeepBelow);
|
|
|
|
connect(this, &AbstractClient::minimizedChanged, w, [w, this] { w->setMinimized(isMinimized()); });
|
|
|
|
connect(this, static_cast<void (AbstractClient::*)(AbstractClient*,MaximizeMode)>(&AbstractClient::clientMaximizedStateChanged), w,
|
|
|
|
[w] (KWin::AbstractClient *c, MaximizeMode mode) {
|
|
|
|
Q_UNUSED(c);
|
|
|
|
w->setMaximized(mode == KWin::MaximizeFull);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(this, &AbstractClient::demandsAttentionChanged, w, [w, this] { w->setDemandsAttention(isDemandingAttention()); });
|
|
|
|
connect(this, &AbstractClient::iconChanged, w,
|
|
|
|
[w, this] {
|
2016-10-13 11:57:34 +00:00
|
|
|
w->setIcon(icon());
|
2015-07-09 07:10:33 +00:00
|
|
|
}
|
|
|
|
);
|
2016-10-27 13:59:08 +00:00
|
|
|
connect(this, &AbstractClient::windowClassChanged, w, updateAppId);
|
|
|
|
connect(this, &AbstractClient::desktopFileNameChanged, w, updateAppId);
|
2016-04-15 11:41:10 +00:00
|
|
|
connect(this, &AbstractClient::shadeChanged, w, [w, this] { w->setShaded(isShade()); });
|
2016-06-06 13:19:28 +00:00
|
|
|
connect(this, &AbstractClient::transientChanged, w,
|
|
|
|
[w, this] {
|
|
|
|
w->setParentWindow(transientFor() ? transientFor()->windowManagementInterface() : nullptr);
|
|
|
|
}
|
|
|
|
);
|
2016-07-18 06:22:33 +00:00
|
|
|
connect(this, &AbstractClient::geometryChanged, w,
|
|
|
|
[w, this] {
|
|
|
|
w->setGeometry(geom);
|
|
|
|
}
|
|
|
|
);
|
2015-07-09 07:10:33 +00:00
|
|
|
connect(w, &PlasmaWindowInterface::closeRequested, this, [this] { closeWindow(); });
|
2016-04-17 14:43:45 +00:00
|
|
|
connect(w, &PlasmaWindowInterface::moveRequested, this,
|
|
|
|
[this] {
|
|
|
|
Cursor::setPos(geometry().center());
|
|
|
|
performMouseCommand(Options::MouseMove, Cursor::pos());
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::resizeRequested, this,
|
|
|
|
[this] {
|
|
|
|
Cursor::setPos(geometry().bottomRight());
|
|
|
|
performMouseCommand(Options::MouseResize, Cursor::pos());
|
|
|
|
}
|
|
|
|
);
|
2015-07-09 07:10:33 +00:00
|
|
|
connect(w, &PlasmaWindowInterface::virtualDesktopRequested, this,
|
|
|
|
[this] (quint32 desktop) {
|
|
|
|
workspace()->sendClientToDesktop(this, desktop + 1, true);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::fullscreenRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
setFullScreen(set, false);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::minimizedRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
if (set) {
|
|
|
|
minimize();
|
|
|
|
} else {
|
|
|
|
unminimize();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::maximizedRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
maximize(set ? MaximizeFull : MaximizeRestore);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::keepAboveRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
setKeepAbove(set);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::keepBelowRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
setKeepBelow(set);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::demandsAttentionRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
demandAttention(set);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(w, &PlasmaWindowInterface::activeRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
if (set) {
|
|
|
|
workspace()->activateClient(this, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2016-04-15 11:41:10 +00:00
|
|
|
connect(w, &PlasmaWindowInterface::shadedRequested, this,
|
|
|
|
[this] (bool set) {
|
|
|
|
setShade(set);
|
|
|
|
}
|
|
|
|
);
|
2015-07-09 07:10:33 +00:00
|
|
|
m_windowManagementInterface = w;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::destroyWindowManagementInterface()
|
|
|
|
{
|
2016-05-12 05:56:46 +00:00
|
|
|
if (m_windowManagementInterface) {
|
|
|
|
m_windowManagementInterface->unmap();
|
|
|
|
m_windowManagementInterface = nullptr;
|
|
|
|
}
|
2015-07-09 07:10:33 +00:00
|
|
|
}
|
|
|
|
|
2015-07-09 14:11:51 +00:00
|
|
|
Options::MouseCommand AbstractClient::getMouseCommand(Qt::MouseButton button, bool *handled) const
|
|
|
|
{
|
|
|
|
*handled = false;
|
|
|
|
if (button == Qt::NoButton) {
|
|
|
|
return Options::MouseNothing;
|
|
|
|
}
|
|
|
|
if (isActive()) {
|
|
|
|
if (options->isClickRaise()) {
|
|
|
|
*handled = true;
|
|
|
|
return Options::MouseActivateRaiseAndPassClick;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
*handled = true;
|
|
|
|
switch (button) {
|
|
|
|
case Qt::LeftButton:
|
|
|
|
return options->commandWindow1();
|
|
|
|
case Qt::MiddleButton:
|
|
|
|
return options->commandWindow2();
|
|
|
|
case Qt::RightButton:
|
|
|
|
return options->commandWindow3();
|
|
|
|
default:
|
|
|
|
// all other buttons pass Activate & Pass Client
|
|
|
|
return Options::MouseActivateAndPassClick;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Options::MouseNothing;
|
|
|
|
}
|
|
|
|
|
|
|
|
Options::MouseCommand AbstractClient::getWheelCommand(Qt::Orientation orientation, bool *handled) const
|
|
|
|
{
|
|
|
|
*handled = false;
|
|
|
|
if (orientation != Qt::Vertical) {
|
|
|
|
return Options::MouseNothing;
|
|
|
|
}
|
|
|
|
if (!isActive()) {
|
|
|
|
*handled = true;
|
|
|
|
return options->commandWindowWheel();
|
|
|
|
}
|
|
|
|
return Options::MouseNothing;
|
|
|
|
}
|
|
|
|
|
2015-07-09 14:08:01 +00:00
|
|
|
bool AbstractClient::performMouseCommand(Options::MouseCommand cmd, const QPoint &globalPos)
|
|
|
|
{
|
|
|
|
bool replay = false;
|
|
|
|
switch(cmd) {
|
|
|
|
case Options::MouseRaise:
|
|
|
|
workspace()->raiseClient(this);
|
|
|
|
break;
|
|
|
|
case Options::MouseLower: {
|
|
|
|
workspace()->lowerClient(this);
|
2016-02-18 09:30:58 +00:00
|
|
|
// used to be activateNextClient(this), then topClientOnDesktop
|
|
|
|
// since this is a mouseOp it's however safe to use the client under the mouse instead
|
|
|
|
if (isActive() && options->focusPolicyIsReasonable()) {
|
|
|
|
AbstractClient *next = workspace()->clientUnderMouse(screen());
|
|
|
|
if (next && next != this)
|
|
|
|
workspace()->requestFocus(next, false);
|
|
|
|
}
|
2015-07-09 14:08:01 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Options::MouseOperationsMenu:
|
|
|
|
if (isActive() && options->isClickRaise())
|
|
|
|
autoRaise();
|
|
|
|
workspace()->showWindowMenu(QRect(globalPos, globalPos), this);
|
|
|
|
break;
|
|
|
|
case Options::MouseToggleRaiseAndLower:
|
|
|
|
workspace()->raiseOrLowerClient(this);
|
|
|
|
break;
|
|
|
|
case Options::MouseActivateAndRaise: {
|
|
|
|
replay = isActive(); // for clickraise mode
|
2015-12-07 15:18:30 +00:00
|
|
|
bool mustReplay = !rules()->checkAcceptFocus(acceptsFocus());
|
2015-07-09 14:08:01 +00:00
|
|
|
if (mustReplay) {
|
|
|
|
ToplevelList::const_iterator it = workspace()->stackingOrder().constEnd(),
|
|
|
|
begin = workspace()->stackingOrder().constBegin();
|
|
|
|
while (mustReplay && --it != begin && *it != this) {
|
|
|
|
AbstractClient *c = qobject_cast<AbstractClient*>(*it);
|
|
|
|
if (!c || (c->keepAbove() && !keepAbove()) || (keepBelow() && !c->keepBelow()))
|
|
|
|
continue; // can never raise above "it"
|
|
|
|
mustReplay = !(c->isOnCurrentDesktop() && c->isOnCurrentActivity() && c->geometry().intersects(geometry()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
workspace()->takeActivity(this, Workspace::ActivityFocus | Workspace::ActivityRaise);
|
|
|
|
screens()->setCurrent(globalPos);
|
|
|
|
replay = replay || mustReplay;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Options::MouseActivateAndLower:
|
|
|
|
workspace()->requestFocus(this);
|
|
|
|
workspace()->lowerClient(this);
|
|
|
|
screens()->setCurrent(globalPos);
|
2015-12-07 15:18:30 +00:00
|
|
|
replay = replay || !rules()->checkAcceptFocus(acceptsFocus());
|
2015-07-09 14:08:01 +00:00
|
|
|
break;
|
|
|
|
case Options::MouseActivate:
|
|
|
|
replay = isActive(); // for clickraise mode
|
|
|
|
workspace()->takeActivity(this, Workspace::ActivityFocus);
|
|
|
|
screens()->setCurrent(globalPos);
|
2015-12-07 15:18:30 +00:00
|
|
|
replay = replay || !rules()->checkAcceptFocus(acceptsFocus());
|
2015-07-09 14:08:01 +00:00
|
|
|
break;
|
|
|
|
case Options::MouseActivateRaiseAndPassClick:
|
|
|
|
workspace()->takeActivity(this, Workspace::ActivityFocus | Workspace::ActivityRaise);
|
|
|
|
screens()->setCurrent(globalPos);
|
|
|
|
replay = true;
|
|
|
|
break;
|
|
|
|
case Options::MouseActivateAndPassClick:
|
|
|
|
workspace()->takeActivity(this, Workspace::ActivityFocus);
|
|
|
|
screens()->setCurrent(globalPos);
|
|
|
|
replay = true;
|
|
|
|
break;
|
|
|
|
case Options::MouseMaximize:
|
|
|
|
maximize(MaximizeFull);
|
|
|
|
break;
|
|
|
|
case Options::MouseRestore:
|
|
|
|
maximize(MaximizeRestore);
|
|
|
|
break;
|
|
|
|
case Options::MouseMinimize:
|
|
|
|
minimize();
|
|
|
|
break;
|
|
|
|
case Options::MouseAbove: {
|
|
|
|
StackingUpdatesBlocker blocker(workspace());
|
|
|
|
if (keepBelow())
|
|
|
|
setKeepBelow(false);
|
|
|
|
else
|
|
|
|
setKeepAbove(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Options::MouseBelow: {
|
|
|
|
StackingUpdatesBlocker blocker(workspace());
|
|
|
|
if (keepAbove())
|
|
|
|
setKeepAbove(false);
|
|
|
|
else
|
|
|
|
setKeepBelow(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Options::MousePreviousDesktop:
|
|
|
|
workspace()->windowToPreviousDesktop(this);
|
|
|
|
break;
|
|
|
|
case Options::MouseNextDesktop:
|
|
|
|
workspace()->windowToNextDesktop(this);
|
|
|
|
break;
|
|
|
|
case Options::MouseOpacityMore:
|
|
|
|
if (!isDesktop()) // No point in changing the opacity of the desktop
|
|
|
|
setOpacity(qMin(opacity() + 0.1, 1.0));
|
|
|
|
break;
|
|
|
|
case Options::MouseOpacityLess:
|
|
|
|
if (!isDesktop()) // No point in changing the opacity of the desktop
|
|
|
|
setOpacity(qMax(opacity() - 0.1, 0.1));
|
|
|
|
break;
|
|
|
|
case Options::MousePreviousTab:
|
|
|
|
if (tabGroup())
|
|
|
|
tabGroup()->activatePrev();
|
|
|
|
break;
|
|
|
|
case Options::MouseNextTab:
|
|
|
|
if (tabGroup())
|
|
|
|
tabGroup()->activateNext();
|
|
|
|
break;
|
|
|
|
case Options::MouseClose:
|
|
|
|
closeWindow();
|
|
|
|
break;
|
2015-10-23 11:40:13 +00:00
|
|
|
case Options::MouseActivateRaiseAndMove:
|
|
|
|
case Options::MouseActivateRaiseAndUnrestrictedMove:
|
|
|
|
workspace()->raiseClient(this);
|
|
|
|
workspace()->requestFocus(this);
|
|
|
|
screens()->setCurrent(globalPos);
|
|
|
|
// fallthrough
|
|
|
|
case Options::MouseMove:
|
|
|
|
case Options::MouseUnrestrictedMove: {
|
|
|
|
if (!isMovableAcrossScreens())
|
|
|
|
break;
|
|
|
|
if (isMoveResize())
|
|
|
|
finishMoveResize(false);
|
|
|
|
setMoveResizePointerMode(PositionCenter);
|
|
|
|
setMoveResizePointerButtonDown(true);
|
|
|
|
setMoveOffset(QPoint(globalPos.x() - x(), globalPos.y() - y())); // map from global
|
|
|
|
setInvertedMoveOffset(rect().bottomRight() - moveOffset());
|
|
|
|
setUnrestrictedMoveResize((cmd == Options::MouseActivateRaiseAndUnrestrictedMove
|
|
|
|
|| cmd == Options::MouseUnrestrictedMove));
|
|
|
|
if (!startMoveResize())
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
updateCursor();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Options::MouseResize:
|
|
|
|
case Options::MouseUnrestrictedResize: {
|
|
|
|
if (!isResizable() || isShade())
|
|
|
|
break;
|
|
|
|
if (isMoveResize())
|
|
|
|
finishMoveResize(false);
|
|
|
|
setMoveResizePointerButtonDown(true);
|
|
|
|
const QPoint moveOffset = QPoint(globalPos.x() - x(), globalPos.y() - y()); // map from global
|
|
|
|
setMoveOffset(moveOffset);
|
|
|
|
int x = moveOffset.x(), y = moveOffset.y();
|
|
|
|
bool left = x < width() / 3;
|
|
|
|
bool right = x >= 2 * width() / 3;
|
|
|
|
bool top = y < height() / 3;
|
|
|
|
bool bot = y >= 2 * height() / 3;
|
|
|
|
Position mode;
|
|
|
|
if (top)
|
|
|
|
mode = left ? PositionTopLeft : (right ? PositionTopRight : PositionTop);
|
|
|
|
else if (bot)
|
|
|
|
mode = left ? PositionBottomLeft : (right ? PositionBottomRight : PositionBottom);
|
|
|
|
else
|
|
|
|
mode = (x < width() / 2) ? PositionLeft : PositionRight;
|
|
|
|
setMoveResizePointerMode(mode);
|
|
|
|
setInvertedMoveOffset(rect().bottomRight() - moveOffset);
|
|
|
|
setUnrestrictedMoveResize((cmd == Options::MouseUnrestrictedResize));
|
|
|
|
if (!startMoveResize())
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
updateCursor();
|
|
|
|
break;
|
|
|
|
}
|
2015-07-09 14:08:01 +00:00
|
|
|
case Options::MouseDragTab:
|
|
|
|
case Options::MouseNothing:
|
|
|
|
default:
|
|
|
|
replay = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return replay;
|
|
|
|
}
|
|
|
|
|
2015-09-11 09:37:40 +00:00
|
|
|
void AbstractClient::setTransientFor(AbstractClient *transientFor)
|
|
|
|
{
|
2015-09-17 12:10:57 +00:00
|
|
|
if (transientFor == this) {
|
|
|
|
// cannot be transient for one self
|
|
|
|
return;
|
|
|
|
}
|
2015-09-11 09:37:40 +00:00
|
|
|
if (m_transientFor == transientFor) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_transientFor = transientFor;
|
|
|
|
emit transientChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
const AbstractClient *AbstractClient::transientFor() const
|
|
|
|
{
|
|
|
|
return m_transientFor;
|
|
|
|
}
|
|
|
|
|
|
|
|
AbstractClient *AbstractClient::transientFor()
|
|
|
|
{
|
|
|
|
return m_transientFor;
|
|
|
|
}
|
|
|
|
|
2015-09-11 11:31:41 +00:00
|
|
|
bool AbstractClient::hasTransientPlacementHint() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint AbstractClient::transientPlacementHint() const
|
|
|
|
{
|
|
|
|
return QPoint();
|
|
|
|
}
|
|
|
|
|
2015-09-11 12:14:21 +00:00
|
|
|
bool AbstractClient::hasTransient(const AbstractClient *c, bool indirect) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(indirect);
|
|
|
|
return c->transientFor() == this;
|
|
|
|
}
|
|
|
|
|
2015-09-11 13:55:23 +00:00
|
|
|
QList< AbstractClient* > AbstractClient::mainClients() const
|
|
|
|
{
|
|
|
|
if (const AbstractClient *t = transientFor()) {
|
|
|
|
return QList<AbstractClient*>{const_cast< AbstractClient* >(t)};
|
|
|
|
}
|
|
|
|
return QList<AbstractClient*>();
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<AbstractClient*> AbstractClient::allMainClients() const
|
|
|
|
{
|
|
|
|
auto result = mainClients();
|
|
|
|
foreach (const auto *cl, result) {
|
|
|
|
result += cl->allMainClients();
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-09-14 07:04:20 +00:00
|
|
|
void AbstractClient::setModal(bool m)
|
|
|
|
{
|
|
|
|
// Qt-3.2 can have even modal normal windows :(
|
|
|
|
if (m_modal == m)
|
|
|
|
return;
|
|
|
|
m_modal = m;
|
|
|
|
emit modalChanged();
|
|
|
|
// Changing modality for a mapped window is weird (?)
|
|
|
|
// _NET_WM_STATE_MODAL should possibly rather be _NET_WM_WINDOW_TYPE_MODAL_DIALOG
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AbstractClient::isModal() const
|
|
|
|
{
|
|
|
|
return m_modal;
|
|
|
|
}
|
|
|
|
|
2015-09-14 08:55:27 +00:00
|
|
|
void AbstractClient::addTransient(AbstractClient *cl)
|
|
|
|
{
|
|
|
|
assert(!m_transients.contains(cl));
|
|
|
|
assert(cl != this);
|
|
|
|
m_transients.append(cl);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::removeTransient(AbstractClient *cl)
|
|
|
|
{
|
|
|
|
m_transients.removeAll(cl);
|
2015-10-01 12:12:46 +00:00
|
|
|
if (cl->transientFor() == this) {
|
|
|
|
cl->setTransientFor(nullptr);
|
|
|
|
}
|
2015-09-14 08:55:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::removeTransientFromList(AbstractClient *cl)
|
|
|
|
{
|
|
|
|
m_transients.removeAll(cl);
|
|
|
|
}
|
|
|
|
|
2015-09-17 09:06:59 +00:00
|
|
|
bool AbstractClient::isActiveFullScreen() const
|
|
|
|
{
|
|
|
|
if (!isFullScreen())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const auto ac = workspace()->mostRecentlyActivatedClient(); // instead of activeClient() - avoids flicker
|
|
|
|
// according to NETWM spec implementation notes suggests
|
|
|
|
// "focused windows having state _NET_WM_STATE_FULLSCREEN" to be on the highest layer.
|
|
|
|
// we'll also take the screen into account
|
2018-01-06 10:24:31 +00:00
|
|
|
return ac && (ac == this || ac->screen() != screen()|| ac->allMainClients().contains(const_cast<AbstractClient*>(this)));
|
2015-09-17 09:06:59 +00:00
|
|
|
}
|
|
|
|
|
2015-12-03 12:54:16 +00:00
|
|
|
#define BORDER(which) \
|
|
|
|
int AbstractClient::border##which() const \
|
|
|
|
{ \
|
|
|
|
return isDecorated() ? decoration()->border##which() : 0; \
|
|
|
|
}
|
2015-10-12 10:00:33 +00:00
|
|
|
|
2015-12-03 12:54:16 +00:00
|
|
|
BORDER(Bottom)
|
|
|
|
BORDER(Left)
|
|
|
|
BORDER(Right)
|
|
|
|
BORDER(Top)
|
|
|
|
#undef BORDER
|
2015-10-12 10:00:33 +00:00
|
|
|
|
2015-10-13 07:49:31 +00:00
|
|
|
QSize AbstractClient::sizeForClientSize(const QSize &wsize, Sizemode mode, bool noframe) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(mode)
|
|
|
|
Q_UNUSED(noframe)
|
2016-11-18 11:32:05 +00:00
|
|
|
return wsize + QSize(borderLeft() + borderRight(), borderTop() + borderBottom());
|
2015-10-13 07:49:31 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 10:50:28 +00:00
|
|
|
void AbstractClient::addRepaintDuringGeometryUpdates()
|
|
|
|
{
|
|
|
|
const QRect deco_rect = visibleRect();
|
|
|
|
addLayerRepaint(m_visibleRectBeforeGeometryUpdate);
|
|
|
|
addLayerRepaint(deco_rect); // trigger repaint of window's new location
|
|
|
|
m_visibleRectBeforeGeometryUpdate = deco_rect;
|
|
|
|
}
|
|
|
|
|
2015-10-15 11:11:21 +00:00
|
|
|
void AbstractClient::updateGeometryBeforeUpdateBlocking()
|
|
|
|
{
|
|
|
|
m_geometryBeforeUpdateBlocking = geom;
|
|
|
|
}
|
|
|
|
|
2015-10-15 11:39:49 +00:00
|
|
|
void AbstractClient::updateTabGroupStates(TabGroup::States)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-10-15 13:28:01 +00:00
|
|
|
void AbstractClient::doMove(int, int)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-10-16 07:57:53 +00:00
|
|
|
void AbstractClient::updateInitialMoveResizeGeometry()
|
|
|
|
{
|
|
|
|
m_moveResize.initialGeometry = geometry();
|
2015-10-16 09:13:41 +00:00
|
|
|
m_moveResize.geometry = m_moveResize.initialGeometry;
|
2015-10-16 13:53:28 +00:00
|
|
|
m_moveResize.startScreen = screen();
|
2015-10-16 07:57:53 +00:00
|
|
|
}
|
|
|
|
|
2015-10-16 13:27:53 +00:00
|
|
|
void AbstractClient::updateCursor()
|
|
|
|
{
|
|
|
|
Position m = moveResizePointerMode();
|
|
|
|
if (!isResizable() || isShade())
|
|
|
|
m = PositionCenter;
|
|
|
|
Qt::CursorShape c = Qt::ArrowCursor;
|
|
|
|
switch(m) {
|
|
|
|
case PositionTopLeft:
|
|
|
|
case PositionBottomRight:
|
|
|
|
c = Qt::SizeFDiagCursor;
|
|
|
|
break;
|
|
|
|
case PositionBottomLeft:
|
|
|
|
case PositionTopRight:
|
|
|
|
c = Qt::SizeBDiagCursor;
|
|
|
|
break;
|
|
|
|
case PositionTop:
|
|
|
|
case PositionBottom:
|
|
|
|
c = Qt::SizeVerCursor;
|
|
|
|
break;
|
|
|
|
case PositionLeft:
|
|
|
|
case PositionRight:
|
|
|
|
c = Qt::SizeHorCursor;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (isMoveResize())
|
|
|
|
c = Qt::SizeAllCursor;
|
|
|
|
else
|
|
|
|
c = Qt::ArrowCursor;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (c == m_moveResize.cursor)
|
|
|
|
return;
|
|
|
|
m_moveResize.cursor = c;
|
|
|
|
emit moveResizeCursorChanged(c);
|
|
|
|
}
|
|
|
|
|
2015-10-22 13:58:24 +00:00
|
|
|
void AbstractClient::leaveMoveResize()
|
|
|
|
{
|
|
|
|
workspace()->setClientIsMoving(nullptr);
|
|
|
|
setMoveResize(false);
|
|
|
|
if (ScreenEdges::self()->isDesktopSwitchingMovingClients())
|
|
|
|
ScreenEdges::self()->reserveDesktopSwitching(false, Qt::Vertical|Qt::Horizontal);
|
|
|
|
if (isElectricBorderMaximizing()) {
|
|
|
|
outline()->hide();
|
|
|
|
elevate(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-22 15:08:17 +00:00
|
|
|
bool AbstractClient::s_haveResizeEffect = false;
|
|
|
|
|
|
|
|
void AbstractClient::updateHaveResizeEffect()
|
|
|
|
{
|
|
|
|
s_haveResizeEffect = effects && static_cast<EffectsHandlerImpl*>(effects)->provides(Effect::Resize);
|
|
|
|
}
|
|
|
|
|
2015-10-22 16:16:21 +00:00
|
|
|
bool AbstractClient::doStartMoveResize()
|
2015-10-22 15:27:05 +00:00
|
|
|
{
|
2015-10-22 16:16:21 +00:00
|
|
|
return true;
|
2015-10-22 15:27:05 +00:00
|
|
|
}
|
|
|
|
|
2015-10-23 10:27:06 +00:00
|
|
|
void AbstractClient::positionGeometryTip()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-10-23 11:05:55 +00:00
|
|
|
void AbstractClient::doPerformMoveResize()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-10-23 11:33:18 +00:00
|
|
|
bool AbstractClient::isWaitingForMoveResizeSync() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doResizeSync()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-10-23 11:32:18 +00:00
|
|
|
void AbstractClient::checkQuickTilingMaximizationZones(int xroot, int yroot)
|
|
|
|
{
|
2017-07-18 19:12:37 +00:00
|
|
|
QuickTileMode mode = QuickTileFlag::None;
|
2015-10-23 11:32:18 +00:00
|
|
|
bool innerBorder = false;
|
|
|
|
for (int i=0; i < screens()->count(); ++i) {
|
|
|
|
|
|
|
|
if (!screens()->geometry(i).contains(QPoint(xroot, yroot)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto isInScreen = [i](const QPoint &pt) {
|
|
|
|
for (int j = 0; j < screens()->count(); ++j) {
|
|
|
|
if (j == i)
|
|
|
|
continue;
|
|
|
|
if (screens()->geometry(j).contains(pt)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
QRect area = workspace()->clientArea(MaximizeArea, QPoint(xroot, yroot), desktop());
|
|
|
|
if (options->electricBorderTiling()) {
|
|
|
|
if (xroot <= area.x() + 20) {
|
2017-07-18 19:12:37 +00:00
|
|
|
mode |= QuickTileFlag::Left;
|
2015-10-23 11:32:18 +00:00
|
|
|
innerBorder = isInScreen(QPoint(area.x() - 1, yroot));
|
|
|
|
} else if (xroot >= area.x() + area.width() - 20) {
|
2017-07-18 19:12:37 +00:00
|
|
|
mode |= QuickTileFlag::Right;
|
2015-10-23 11:32:18 +00:00
|
|
|
innerBorder = isInScreen(QPoint(area.right() + 1, yroot));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-18 19:12:37 +00:00
|
|
|
if (mode != QuickTileMode(QuickTileFlag::None)) {
|
2015-10-23 11:32:18 +00:00
|
|
|
if (yroot <= area.y() + area.height() * options->electricBorderCornerRatio())
|
2017-07-18 19:12:37 +00:00
|
|
|
mode |= QuickTileFlag::Top;
|
2015-10-23 11:32:18 +00:00
|
|
|
else if (yroot >= area.y() + area.height() - area.height() * options->electricBorderCornerRatio())
|
2017-07-18 19:12:37 +00:00
|
|
|
mode |= QuickTileFlag::Bottom;
|
2015-10-23 11:32:18 +00:00
|
|
|
} else if (options->electricBorderMaximize() && yroot <= area.y() + 5 && isMaximizable()) {
|
2017-07-18 19:12:37 +00:00
|
|
|
mode = QuickTileFlag::Maximize;
|
2015-10-23 11:32:18 +00:00
|
|
|
innerBorder = isInScreen(QPoint(xroot, area.y() - 1));
|
|
|
|
}
|
|
|
|
break; // no point in checking other screens to contain this... "point"...
|
|
|
|
}
|
|
|
|
if (mode != electricBorderMode()) {
|
|
|
|
setElectricBorderMode(mode);
|
|
|
|
if (innerBorder) {
|
|
|
|
if (!m_electricMaximizingDelay) {
|
|
|
|
m_electricMaximizingDelay = new QTimer(this);
|
|
|
|
m_electricMaximizingDelay->setInterval(250);
|
|
|
|
m_electricMaximizingDelay->setSingleShot(true);
|
|
|
|
connect(m_electricMaximizingDelay, &QTimer::timeout, [this]() {
|
|
|
|
if (isMove())
|
2017-07-18 19:12:37 +00:00
|
|
|
setElectricBorderMaximizing(electricBorderMode() != QuickTileMode(QuickTileFlag::None));
|
2015-10-23 11:32:18 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
m_electricMaximizingDelay->start();
|
|
|
|
} else {
|
2017-07-18 19:12:37 +00:00
|
|
|
setElectricBorderMaximizing(mode != QuickTileMode(QuickTileFlag::None));
|
2015-10-23 11:32:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-23 13:32:10 +00:00
|
|
|
void AbstractClient::keyPressEvent(uint key_code)
|
|
|
|
{
|
|
|
|
if (!isMove() && !isResize())
|
|
|
|
return;
|
|
|
|
bool is_control = key_code & Qt::CTRL;
|
|
|
|
bool is_alt = key_code & Qt::ALT;
|
|
|
|
key_code = key_code & ~Qt::KeyboardModifierMask;
|
|
|
|
int delta = is_control ? 1 : is_alt ? 32 : 8;
|
|
|
|
QPoint pos = Cursor::pos();
|
|
|
|
switch(key_code) {
|
|
|
|
case Qt::Key_Left:
|
|
|
|
pos.rx() -= delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Right:
|
|
|
|
pos.rx() += delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Up:
|
|
|
|
pos.ry() -= delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Down:
|
|
|
|
pos.ry() += delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Space:
|
|
|
|
case Qt::Key_Return:
|
|
|
|
case Qt::Key_Enter:
|
|
|
|
finishMoveResize(false);
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
updateCursor();
|
|
|
|
break;
|
|
|
|
case Qt::Key_Escape:
|
|
|
|
finishMoveResize(true);
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
updateCursor();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Cursor::setPos(pos);
|
|
|
|
}
|
|
|
|
|
2015-10-26 14:49:03 +00:00
|
|
|
QSize AbstractClient::resizeIncrements() const
|
|
|
|
{
|
|
|
|
return QSize(1, 1);
|
|
|
|
}
|
|
|
|
|
2015-12-03 14:27:21 +00:00
|
|
|
void AbstractClient::dontMoveResize()
|
|
|
|
{
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
stopDelayedMoveResize();
|
|
|
|
if (isMoveResize())
|
|
|
|
finishMoveResize(false);
|
|
|
|
}
|
|
|
|
|
2015-10-28 09:43:49 +00:00
|
|
|
AbstractClient::Position AbstractClient::mousePosition() const
|
|
|
|
{
|
2015-12-03 13:07:32 +00:00
|
|
|
if (isDecorated()) {
|
|
|
|
switch (decoration()->sectionUnderMouse()) {
|
|
|
|
case Qt::BottomLeftSection:
|
|
|
|
return PositionBottomLeft;
|
|
|
|
case Qt::BottomRightSection:
|
|
|
|
return PositionBottomRight;
|
|
|
|
case Qt::BottomSection:
|
|
|
|
return PositionBottom;
|
|
|
|
case Qt::LeftSection:
|
|
|
|
return PositionLeft;
|
|
|
|
case Qt::RightSection:
|
|
|
|
return PositionRight;
|
|
|
|
case Qt::TopSection:
|
|
|
|
return PositionTop;
|
|
|
|
case Qt::TopLeftSection:
|
|
|
|
return PositionTopLeft;
|
|
|
|
case Qt::TopRightSection:
|
|
|
|
return PositionTopRight;
|
|
|
|
default:
|
|
|
|
return PositionCenter;
|
|
|
|
}
|
|
|
|
}
|
2015-10-28 09:43:49 +00:00
|
|
|
return PositionCenter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::endMoveResize()
|
|
|
|
{
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
stopDelayedMoveResize();
|
|
|
|
if (isMoveResize()) {
|
|
|
|
finishMoveResize(false);
|
|
|
|
setMoveResizePointerMode(mousePosition());
|
|
|
|
}
|
|
|
|
updateCursor();
|
|
|
|
}
|
|
|
|
|
2015-12-03 12:24:44 +00:00
|
|
|
void AbstractClient::destroyDecoration()
|
|
|
|
{
|
2015-12-04 07:12:49 +00:00
|
|
|
delete m_decoration.decoration;
|
|
|
|
m_decoration.decoration = nullptr;
|
2015-12-03 12:24:44 +00:00
|
|
|
}
|
|
|
|
|
2015-12-03 12:28:22 +00:00
|
|
|
bool AbstractClient::decorationHasAlpha() const
|
|
|
|
{
|
|
|
|
if (!isDecorated() || decoration()->isOpaque()) {
|
|
|
|
// either no decoration or decoration has alpha disabled
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-03 12:31:25 +00:00
|
|
|
void AbstractClient::triggerDecorationRepaint()
|
|
|
|
{
|
|
|
|
if (isDecorated()) {
|
|
|
|
decoration()->update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-03 12:48:54 +00:00
|
|
|
void AbstractClient::layoutDecorationRects(QRect &left, QRect &top, QRect &right, QRect &bottom) const
|
|
|
|
{
|
|
|
|
if (!isDecorated()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QRect r = decoration()->rect();
|
|
|
|
|
|
|
|
top = QRect(r.x(), r.y(), r.width(), borderTop());
|
|
|
|
bottom = QRect(r.x(), r.y() + r.height() - borderBottom(),
|
|
|
|
r.width(), borderBottom());
|
|
|
|
left = QRect(r.x(), r.y() + top.height(),
|
|
|
|
borderLeft(), r.height() - top.height() - bottom.height());
|
|
|
|
right = QRect(r.x() + r.width() - borderRight(), r.y() + top.height(),
|
|
|
|
borderRight(), r.height() - top.height() - bottom.height());
|
|
|
|
}
|
|
|
|
|
2016-03-11 11:48:01 +00:00
|
|
|
void AbstractClient::processDecorationMove(const QPoint &localPos, const QPoint &globalPos)
|
2015-12-03 13:10:12 +00:00
|
|
|
{
|
|
|
|
if (isMoveResizePointerButtonDown()) {
|
2016-03-11 11:48:01 +00:00
|
|
|
handleMoveResize(localPos.x(), localPos.y(), globalPos.x(), globalPos.y());
|
2015-12-03 13:10:12 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO: handle modifiers
|
|
|
|
Position newmode = mousePosition();
|
|
|
|
if (newmode != moveResizePointerMode()) {
|
|
|
|
setMoveResizePointerMode(newmode);
|
|
|
|
updateCursor();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-03 15:40:27 +00:00
|
|
|
bool AbstractClient::processDecorationButtonPress(QMouseEvent *event, bool ignoreMenu)
|
|
|
|
{
|
|
|
|
Options::MouseCommand com = Options::MouseNothing;
|
|
|
|
bool active = isActive();
|
|
|
|
if (!wantsInput()) // we cannot be active, use it anyway
|
|
|
|
active = true;
|
|
|
|
|
|
|
|
// check whether it is a double click
|
2016-05-12 08:28:40 +00:00
|
|
|
if (event->button() == Qt::LeftButton && titlebarPositionUnderMouse()) {
|
2016-01-05 10:43:12 +00:00
|
|
|
if (m_decoration.doubleClickTimer.isValid()) {
|
2016-06-29 08:52:43 +00:00
|
|
|
const qint64 interval = m_decoration.doubleClickTimer.elapsed();
|
2015-12-04 07:12:49 +00:00
|
|
|
m_decoration.doubleClickTimer.invalidate();
|
2016-01-05 10:43:12 +00:00
|
|
|
if (interval > QGuiApplication::styleHints()->mouseDoubleClickInterval()) {
|
2017-06-22 04:52:57 +00:00
|
|
|
m_decoration.doubleClickTimer.start(); // expired -> new first click and pot. init
|
2016-01-05 10:43:12 +00:00
|
|
|
} else {
|
|
|
|
Workspace::self()->performWindowOperation(this, options->operationTitlebarDblClick());
|
|
|
|
dontMoveResize();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
m_decoration.doubleClickTimer.start(); // new first click and pot. init, could be invalidated by release - see below
|
2015-12-03 15:40:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event->button() == Qt::LeftButton)
|
|
|
|
com = active ? options->commandActiveTitlebar1() : options->commandInactiveTitlebar1();
|
|
|
|
else if (event->button() == Qt::MidButton)
|
|
|
|
com = active ? options->commandActiveTitlebar2() : options->commandInactiveTitlebar2();
|
|
|
|
else if (event->button() == Qt::RightButton)
|
|
|
|
com = active ? options->commandActiveTitlebar3() : options->commandInactiveTitlebar3();
|
|
|
|
if (event->button() == Qt::LeftButton
|
|
|
|
&& com != Options::MouseOperationsMenu // actions where it's not possible to get the matching
|
|
|
|
&& com != Options::MouseMinimize // mouse release event
|
|
|
|
&& com != Options::MouseDragTab) {
|
|
|
|
setMoveResizePointerMode(mousePosition());
|
|
|
|
setMoveResizePointerButtonDown(true);
|
|
|
|
setMoveOffset(event->pos());
|
|
|
|
setInvertedMoveOffset(rect().bottomRight() - moveOffset());
|
|
|
|
setUnrestrictedMoveResize(false);
|
|
|
|
startDelayedMoveResize();
|
|
|
|
updateCursor();
|
|
|
|
}
|
|
|
|
// In the new API the decoration may process the menu action to display an inactive tab's menu.
|
|
|
|
// If the event is unhandled then the core will create one for the active window in the group.
|
|
|
|
if (!ignoreMenu || com != Options::MouseOperationsMenu)
|
|
|
|
performMouseCommand(com, event->globalPos());
|
|
|
|
return !( // Return events that should be passed to the decoration in the new API
|
|
|
|
com == Options::MouseRaise ||
|
|
|
|
com == Options::MouseOperationsMenu ||
|
|
|
|
com == Options::MouseActivateAndRaise ||
|
|
|
|
com == Options::MouseActivate ||
|
|
|
|
com == Options::MouseActivateRaiseAndPassClick ||
|
|
|
|
com == Options::MouseActivateAndPassClick ||
|
|
|
|
com == Options::MouseDragTab ||
|
|
|
|
com == Options::MouseNothing);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::processDecorationButtonRelease(QMouseEvent *event)
|
|
|
|
{
|
|
|
|
if (isDecorated()) {
|
2016-05-12 08:28:40 +00:00
|
|
|
if (event->isAccepted() || !titlebarPositionUnderMouse()) {
|
2016-02-17 13:41:26 +00:00
|
|
|
invalidateDecorationDoubleClickTimer(); // click was for the deco and shall not init a doubleclick
|
2015-12-03 15:40:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event->buttons() == Qt::NoButton) {
|
|
|
|
setMoveResizePointerButtonDown(false);
|
|
|
|
stopDelayedMoveResize();
|
|
|
|
if (isMoveResize()) {
|
|
|
|
finishMoveResize(false);
|
|
|
|
setMoveResizePointerMode(mousePosition());
|
|
|
|
}
|
|
|
|
updateCursor();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AbstractClient::startDecorationDoubleClickTimer()
|
|
|
|
{
|
2015-12-04 07:12:49 +00:00
|
|
|
m_decoration.doubleClickTimer.start();
|
2015-12-03 15:40:27 +00:00
|
|
|
}
|
|
|
|
|
2016-01-05 10:43:12 +00:00
|
|
|
void AbstractClient::invalidateDecorationDoubleClickTimer()
|
|
|
|
{
|
|
|
|
m_decoration.doubleClickTimer.invalidate();
|
|
|
|
}
|
|
|
|
|
2015-12-03 16:11:14 +00:00
|
|
|
bool AbstractClient::providesContextHelp() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::showContextHelp()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-12-03 16:27:00 +00:00
|
|
|
QPointer<Decoration::DecoratedClientImpl> AbstractClient::decoratedClient() const
|
|
|
|
{
|
2015-12-04 07:12:49 +00:00
|
|
|
return m_decoration.client;
|
2015-12-03 16:27:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setDecoratedClient(QPointer< Decoration::DecoratedClientImpl > client)
|
|
|
|
{
|
2015-12-04 07:12:49 +00:00
|
|
|
m_decoration.client = client;
|
2015-12-03 16:27:00 +00:00
|
|
|
}
|
|
|
|
|
2016-02-18 10:30:52 +00:00
|
|
|
void AbstractClient::enterEvent(const QPoint &globalPos)
|
|
|
|
{
|
|
|
|
// TODO: shade hover
|
|
|
|
if (options->focusPolicy() == Options::ClickToFocus || workspace()->userActionsMenu()->isShown())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (options->isAutoRaise() && !isDesktop() &&
|
|
|
|
!isDock() && workspace()->focusChangeEnabled() &&
|
|
|
|
globalPos != workspace()->focusMousePosition() &&
|
|
|
|
workspace()->topClientOnDesktop(VirtualDesktopManager::self()->current(),
|
|
|
|
options->isSeparateScreenFocus() ? screen() : -1) != this) {
|
|
|
|
startAutoRaise();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isDesktop() || isDock())
|
|
|
|
return;
|
|
|
|
// for FocusFollowsMouse, change focus only if the mouse has actually been moved, not if the focus
|
|
|
|
// change came because of window changes (e.g. closing a window) - #92290
|
|
|
|
if (options->focusPolicy() != Options::FocusFollowsMouse
|
|
|
|
|| globalPos != workspace()->focusMousePosition()) {
|
|
|
|
workspace()->requestDelayFocus(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-18 12:00:50 +00:00
|
|
|
void AbstractClient::leaveEvent()
|
|
|
|
{
|
|
|
|
cancelAutoRaise();
|
|
|
|
workspace()->cancelDelayFocus();
|
|
|
|
// TODO: shade hover
|
|
|
|
// TODO: send hover leave to deco
|
|
|
|
// TODO: handle Options::FocusStrictlyUnderMouse
|
|
|
|
}
|
|
|
|
|
2016-08-22 14:55:48 +00:00
|
|
|
QRect AbstractClient::iconGeometry() const
|
|
|
|
{
|
|
|
|
if (!windowManagementInterface() || !waylandServer()) {
|
|
|
|
// window management interface is only available if the surface is mapped
|
|
|
|
return QRect();
|
|
|
|
}
|
|
|
|
|
|
|
|
int minDistance = INT_MAX;
|
|
|
|
AbstractClient *candidatePanel = nullptr;
|
|
|
|
QRect candidateGeom;
|
|
|
|
|
|
|
|
for (auto i = windowManagementInterface()->minimizedGeometries().constBegin(), end = windowManagementInterface()->minimizedGeometries().constEnd(); i != end; ++i) {
|
|
|
|
AbstractClient *client = waylandServer()->findAbstractClient(i.key());
|
|
|
|
if (!client) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const int distance = QPoint(client->pos() - pos()).manhattanLength();
|
|
|
|
if (distance < minDistance) {
|
|
|
|
minDistance = distance;
|
|
|
|
candidatePanel = client;
|
|
|
|
candidateGeom = i.value();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!candidatePanel) {
|
|
|
|
return QRect();
|
|
|
|
}
|
|
|
|
return candidateGeom.translated(candidatePanel->pos());
|
|
|
|
}
|
|
|
|
|
2016-09-15 09:44:32 +00:00
|
|
|
QRect AbstractClient::inputGeometry() const
|
|
|
|
{
|
|
|
|
if (isDecorated()) {
|
|
|
|
return Toplevel::inputGeometry() + decoration()->resizeOnlyBorders();
|
|
|
|
}
|
|
|
|
return Toplevel::inputGeometry();
|
|
|
|
}
|
|
|
|
|
2016-10-12 13:09:52 +00:00
|
|
|
bool AbstractClient::dockWantsInput() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-11 16:13:32 +00:00
|
|
|
void AbstractClient::setDesktopFileName(QByteArray name)
|
2016-10-27 13:59:08 +00:00
|
|
|
{
|
2018-03-11 16:13:32 +00:00
|
|
|
name = rules()->checkDesktopFile(name).toUtf8();
|
2016-10-27 13:59:08 +00:00
|
|
|
if (name == m_desktopFileName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_desktopFileName = name;
|
2018-03-11 16:13:32 +00:00
|
|
|
updateWindowRules(Rules::DesktopFile);
|
2016-10-27 13:59:08 +00:00
|
|
|
emit desktopFileNameChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString AbstractClient::iconFromDesktopFile() const
|
|
|
|
{
|
|
|
|
if (m_desktopFileName.isEmpty()) {
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
QString desktopFile = QString::fromUtf8(m_desktopFileName);
|
|
|
|
if (!desktopFile.endsWith(QLatin1String(".desktop"))) {
|
|
|
|
desktopFile.append(QLatin1String(".desktop"));
|
|
|
|
}
|
|
|
|
KDesktopFile df(desktopFile);
|
|
|
|
return df.readIcon();
|
|
|
|
}
|
|
|
|
|
2017-01-11 09:21:03 +00:00
|
|
|
bool AbstractClient::hasApplicationMenu() const
|
|
|
|
{
|
|
|
|
return ApplicationMenu::self()->applicationMenuEnabled() && !m_applicationMenuServiceName.isEmpty() && !m_applicationMenuObjectPath.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::updateApplicationMenuServiceName(const QString &serviceName)
|
|
|
|
{
|
|
|
|
const bool old_hasApplicationMenu = hasApplicationMenu();
|
|
|
|
|
|
|
|
m_applicationMenuServiceName = serviceName;
|
|
|
|
|
|
|
|
const bool new_hasApplicationMenu = hasApplicationMenu();
|
|
|
|
|
|
|
|
if (old_hasApplicationMenu != new_hasApplicationMenu) {
|
|
|
|
emit hasApplicationMenuChanged(new_hasApplicationMenu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::updateApplicationMenuObjectPath(const QString &objectPath)
|
|
|
|
{
|
|
|
|
const bool old_hasApplicationMenu = hasApplicationMenu();
|
|
|
|
|
|
|
|
m_applicationMenuObjectPath = objectPath;
|
|
|
|
|
|
|
|
const bool new_hasApplicationMenu = hasApplicationMenu();
|
|
|
|
|
|
|
|
if (old_hasApplicationMenu != new_hasApplicationMenu) {
|
|
|
|
emit hasApplicationMenuChanged(new_hasApplicationMenu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setApplicationMenuActive(bool applicationMenuActive)
|
|
|
|
{
|
|
|
|
if (m_applicationMenuActive != applicationMenuActive) {
|
|
|
|
m_applicationMenuActive = applicationMenuActive;
|
|
|
|
emit applicationMenuActiveChanged(applicationMenuActive);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::showApplicationMenu(int actionId)
|
|
|
|
{
|
|
|
|
if (isDecorated()) {
|
|
|
|
decoration()->showApplicationMenu(actionId);
|
|
|
|
} else {
|
|
|
|
// we don't know where the application menu button will be, show it in the top left corner instead
|
|
|
|
Workspace::self()->showApplicationMenu(QRect(), this, actionId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-05 09:16:23 +00:00
|
|
|
bool AbstractClient::unresponsive() const
|
|
|
|
{
|
|
|
|
return m_unresponsive;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setUnresponsive(bool unresponsive)
|
|
|
|
{
|
|
|
|
if (m_unresponsive != unresponsive) {
|
|
|
|
m_unresponsive = unresponsive;
|
|
|
|
emit unresponsiveChanged(m_unresponsive);
|
|
|
|
emit captionChanged();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-03 05:11:40 +00:00
|
|
|
QString AbstractClient::shortcutCaptionSuffix() const
|
|
|
|
{
|
|
|
|
if (shortcut().isEmpty()) {
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
return QLatin1String(" {") + shortcut().toString() + QLatin1Char('}');
|
|
|
|
}
|
|
|
|
|
2017-08-20 06:56:13 +00:00
|
|
|
AbstractClient *AbstractClient::findClientWithSameCaption() const
|
|
|
|
{
|
|
|
|
auto fetchNameInternalPredicate = [this](const AbstractClient *cl) {
|
|
|
|
return (!cl->isSpecialWindow() || cl->isToolbar()) && cl != this && cl->captionNormal() == captionNormal() && cl->captionSuffix() == captionSuffix();
|
|
|
|
};
|
|
|
|
return workspace()->findAbstractClient(fetchNameInternalPredicate);
|
|
|
|
}
|
|
|
|
|
2017-08-20 07:35:15 +00:00
|
|
|
QString AbstractClient::caption() const
|
|
|
|
{
|
|
|
|
QString cap = captionNormal() + captionSuffix();
|
|
|
|
if (unresponsive()) {
|
|
|
|
cap += QLatin1String(" ");
|
|
|
|
cap += i18nc("Application is not responding, appended to window title", "(Not Responding)");
|
2017-08-20 07:24:10 +00:00
|
|
|
}
|
|
|
|
return cap;
|
|
|
|
}
|
|
|
|
|
2017-10-01 14:38:57 +00:00
|
|
|
void AbstractClient::removeRule(Rules* rule)
|
|
|
|
{
|
|
|
|
m_rules.remove(rule);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::discardTemporaryRules()
|
|
|
|
{
|
|
|
|
m_rules.discardTemporary();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::evaluateWindowRules()
|
|
|
|
{
|
|
|
|
setupWindowRules(true);
|
|
|
|
applyWindowRules();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::setOnActivities(QStringList newActivitiesList)
|
|
|
|
{
|
|
|
|
Q_UNUSED(newActivitiesList)
|
|
|
|
}
|
|
|
|
|
2017-10-01 19:48:57 +00:00
|
|
|
void AbstractClient::checkNoBorder()
|
|
|
|
{
|
|
|
|
setNoBorder(false);
|
|
|
|
}
|
|
|
|
|
2015-03-05 09:21:03 +00:00
|
|
|
}
|