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"
|
2015-04-29 10:02:54 +00:00
|
|
|
#include "decorations/decorationpalette.h"
|
2015-03-13 11:54:11 +00:00
|
|
|
#include "focuschain.h"
|
2015-03-12 14:21:07 +00:00
|
|
|
#ifdef KWIN_BUILD_TABBOX
|
|
|
|
#include "tabbox.h"
|
|
|
|
#endif
|
2015-03-12 15:08:19 +00:00
|
|
|
#include "workspace.h"
|
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
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
AbstractClient::~AbstractClient() = default;
|
|
|
|
|
|
|
|
void AbstractClient::updateMouseGrab()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AbstractClient::belongToSameApplication(const AbstractClient *c1, const AbstractClient *c2, bool active_hack)
|
|
|
|
{
|
|
|
|
return c1->belongsToSameApplication(c2, active_hack);
|
|
|
|
}
|
|
|
|
|
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 09:24:38 +00:00
|
|
|
void AbstractClient::growHorizontal()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::growVertical()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::shrinkHorizontal()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::shrinkVertical()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-03-12 10:02:09 +00:00
|
|
|
void AbstractClient::packTo(int left, int top)
|
|
|
|
{
|
|
|
|
Q_UNUSED(left)
|
|
|
|
Q_UNUSED(top)
|
|
|
|
}
|
|
|
|
|
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-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);
|
|
|
|
|
|
|
|
doSetActive();
|
|
|
|
emit activeChanged();
|
|
|
|
updateMouseGrab();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AbstractClient::doSetActive()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-03-13 08:26:09 +00:00
|
|
|
void AbstractClient::updateLayer()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
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-03-05 09:21:03 +00:00
|
|
|
}
|