1
0
mirror of https://github.com/KDE/latte-dock.git synced 2025-01-28 10:03:52 +03:00
latte-dock/app/waylandinterface.cpp

281 lines
8.2 KiB
C++
Raw Normal View History

/*
* Copyright 2016 Smith AR <audoban@openmailbox.org>
* Michail Vourlakos <mvourlakos@gmail.com>
*
* This file is part of Latte-Dock
*
* Latte-Dock 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.
*
* Latte-Dock 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 "waylandinterface.h"
#include "../liblattedock/extras.h"
#include <QDebug>
#include <QTimer>
2017-06-10 16:33:46 -05:00
#include <QApplication>
#include <QSignalMapper>
#include <QtX11Extras/QX11Info>
#include <KWindowSystem>
#include <KWindowInfo>
#include <NETWM>
2017-06-10 16:33:46 -05:00
using namespace KWayland::Client;
namespace Latte {
WaylandInterface::WaylandInterface(QObject *parent)
: AbstractWindowInterface(parent)
{
m_activities = new KActivities::Consumer(this);
2017-06-10 16:33:46 -05:00
m_connection = ConnectionThread::fromApplication(this);
if (!m_connection) {
qWarning() << "Failed getting Wayland connection from QPA";
return;
}
m_registry = new Registry(this);
m_registry->create(m_connection);
connect(qApp, &QCoreApplication::aboutToQuit, this, [&]() {
2017-06-10 16:33:46 -05:00
if (m_wm)
m_wm->release();
if (m_plasmaShell)
m_plasmaShell->release();
m_registry->release();
});
2017-06-10 16:33:46 -05:00
m_registry->setup();
m_connection->roundtrip();
2017-06-23 16:00:16 -05:00
const auto wmInterface = m_registry->interface(Registry::Interface::PlasmaWindowManagement);
2017-06-10 16:33:46 -05:00
if (wmInterface.name == 0) {
qWarning() << "This compositor does not support the Plasma Window Management interface";
return;
}
m_wm = m_registry->createPlasmaWindowManagement(wmInterface.name, wmInterface.version, this);
connect(m_wm, &PlasmaWindowManagement::windowCreated, this, &WaylandInterface::windowCreatedProxy);
connect(m_wm, &PlasmaWindowManagement::activeWindowChanged, this, [&]() {
2017-06-23 16:10:14 -05:00
auto w = m_wm->activeWindow();
2017-06-14 16:10:31 -05:00
emit activeWindowChanged(w ? w->internalId() : 0);
2017-06-14 12:50:17 -05:00
}, Qt::QueuedConnection);
2017-06-10 16:33:46 -05:00
2017-06-23 16:00:16 -05:00
const auto shellInterface = m_registry->interface(Registry::Interface::PlasmaShell);
2017-06-10 16:33:46 -05:00
if (shellInterface.name == 0) {
qWarning() << "Plasma Shell interface can't be created";
return;
}
m_plasmaShell = m_registry->createPlasmaShell(shellInterface.name, shellInterface.version, this);
connect(m_activities.data(), &KActivities::Consumer::currentActivityChanged
, this, &WaylandInterface::currentActivityChanged);
}
WaylandInterface::~WaylandInterface()
{
}
2017-06-10 16:33:46 -05:00
void WaylandInterface::init()
{
}
void WaylandInterface::setDockExtraFlags(QWindow &view)
{
Q_UNUSED(view)
}
2017-06-20 19:07:47 -05:00
void WaylandInterface::setDockStruts(QWindow &view, const QRect &dockRect , Plasma::Types::Location location)
{
2017-06-20 19:07:47 -05:00
//TODO: implement AlwaysVisible
Q_UNUSED(view)
Q_UNUSED(dockRect)
Q_UNUSED(location)
}
void WaylandInterface::removeDockStruts(QWindow &view) const
{
KWindowSystem::setStrut(view.winId(), 0, 0, 0, 0);
}
WindowId WaylandInterface::activeWindow() const
{
2017-06-23 16:11:43 -05:00
auto wid = m_wm->activeWindow();
2017-06-10 16:33:46 -05:00
return wid ? wid->internalId() : 0;
}
const std::list<WindowId> &WaylandInterface::windows() const
{
return m_windows;
}
void WaylandInterface::skipTaskBar(const QDialog &dialog) const
{
KWindowSystem::setState(dialog.winId(), NET::SkipTaskbar);
}
void WaylandInterface::slideWindow(QWindow &view, AbstractWindowInterface::Slide location) const
{
auto slideLocation = KWindowEffects::NoEdge;
switch (location) {
case Slide::Top:
slideLocation = KWindowEffects::TopEdge;
break;
case Slide::Bottom:
slideLocation = KWindowEffects::BottomEdge;
break;
case Slide::Left:
slideLocation = KWindowEffects::LeftEdge;
break;
case Slide::Right:
slideLocation = KWindowEffects::RightEdge;
break;
default:
break;
}
KWindowEffects::slideWindow(view.winId(), slideLocation, -1);
}
void WaylandInterface::enableBlurBehind(QWindow &view) const
{
KWindowEffects::enableBlurBehind(view.winId());
}
WindowInfoWrap WaylandInterface::requestInfoActive() const
{
2017-06-23 16:00:16 -05:00
auto w = m_wm->activeWindow();
2017-06-10 16:33:46 -05:00
if (!w) return {};
WindowInfoWrap winfoWrap;
winfoWrap.setIsValid(true);
winfoWrap.setWid(w->internalId());
winfoWrap.setIsActive(w->isActive());
2017-06-14 12:50:17 -05:00
winfoWrap.setIsMinimized(w->isMinimized());
2017-06-10 16:33:46 -05:00
winfoWrap.setIsMaxVert(w->isMaximized());
winfoWrap.setIsMaxHoriz(w->isMaximized());
winfoWrap.setIsFullscreen(w->isFullscreen());
winfoWrap.setIsShaded(w->isShaded());
2017-06-14 12:50:17 -05:00
winfoWrap.setGeometry(w->geometry());
2017-06-10 16:33:46 -05:00
return winfoWrap;
}
bool WaylandInterface::isOnCurrentDesktop(WindowId wid) const
{
2017-06-10 16:33:46 -05:00
auto it = std::find_if(m_wm->windows().constBegin(), m_wm->windows().constEnd(), [&wid](PlasmaWindow * w){
return w->isValid() && w->internalId() == wid;
});
2017-06-14 14:45:43 -05:00
//qDebug() << "desktop:" << (it != m_wm->windows().constEnd() ? (*it)->virtualDesktop() : -1) << KWindowSystem::currentDesktop();
//return true;
2017-06-10 16:33:46 -05:00
return it != m_wm->windows().constEnd() && ( (*it)->virtualDesktop() == KWindowSystem::currentDesktop() || (*it)->isOnAllDesktops());
}
2017-06-20 19:07:47 -05:00
bool WaylandInterface::isOnCurrentActivity(WindowId wid) const
{
auto it = std::find_if(m_wm->windows().constBegin(), m_wm->windows().constEnd(), [&wid](PlasmaWindow * w){
return w->isValid() && w->internalId() == wid;
});
//TODO: Not yet implemented
return it != m_wm->windows().constEnd() && true;
}
WindowInfoWrap WaylandInterface::requestInfo(WindowId wid) const
{
WindowInfoWrap winfoWrap;
2017-06-10 16:33:46 -05:00
auto it = std::find_if(m_wm->windows().constBegin(), m_wm->windows().constEnd(), [&wid](PlasmaWindow * w){
return w->isValid() && w->internalId() == wid;
});
2017-06-10 16:33:46 -05:00
if (it != m_wm->windows().constEnd()) {
auto w = *it;
winfoWrap.setIsValid(true);
winfoWrap.setWid(wid);
winfoWrap.setIsActive(w->isActive());
2017-06-14 12:50:17 -05:00
winfoWrap.setIsMinimized(w->isMinimized());
2017-06-10 16:33:46 -05:00
winfoWrap.setIsMaxVert(w->isMaximized());
winfoWrap.setIsMaxHoriz(w->isMaximized());
winfoWrap.setIsFullscreen(w->isFullscreen());
winfoWrap.setIsShaded(w->isShaded());
2017-06-14 12:50:17 -05:00
winfoWrap.setGeometry(w->geometry());
2017-06-10 16:33:46 -05:00
} else {
winfoWrap.setIsValid(false);
winfoWrap.setWid(wid);
}
return winfoWrap;
}
2017-06-10 16:33:46 -05:00
inline bool WaylandInterface::isValidWindow(const KWayland::Client::PlasmaWindow *w) const
{
2017-06-10 16:33:46 -05:00
return w->isValid() && !w->skipTaskbar();
}
2017-06-10 16:33:46 -05:00
void WaylandInterface::windowCreatedProxy(KWayland::Client::PlasmaWindow *w)
{
2017-06-10 16:33:46 -05:00
if (!isValidWindow(w)) return;
2017-06-10 16:33:46 -05:00
if (!mapper) mapper = new QSignalMapper(this);
2017-06-10 16:33:46 -05:00
mapper->setMapping(w, w);
2017-06-10 16:33:46 -05:00
connect(w, &PlasmaWindow::unmapped, this, [&, win = w]() {
mapper->removeMappings(win);
m_windows.remove(win->internalId());
emit windowRemoved(win->internalId());
});
connect(w, SIGNAL(activeChanged()), mapper, SLOT(map()));
2017-06-10 16:33:46 -05:00
connect(w, SIGNAL(fullscreenChanged()), mapper, SLOT(map()));
connect(w, SIGNAL(geometryChanged()), mapper, SLOT(map()));
connect(w, SIGNAL(maximizedChanged()), mapper, SLOT(map()));
2017-06-14 14:45:43 -05:00
connect(w, SIGNAL(minimizedChanged()), mapper, SLOT(map()));
2017-06-10 16:33:46 -05:00
connect(w, SIGNAL(shadedChanged()), mapper, SLOT(map()));
2017-06-14 16:10:31 -05:00
connect(w, SIGNAL(skipTaskbarChanged()), mapper, SLOT(map()));
2017-06-14 14:45:43 -05:00
connect(w, SIGNAL(onAllDesktopsChanged()), mapper, SLOT(map()));
2017-06-10 16:33:46 -05:00
connect(w, SIGNAL(virtualDesktopChanged()), mapper, SLOT(map()));
connect(mapper, static_cast<void (QSignalMapper::*)(QObject *)>(&QSignalMapper::mapped)
, this, [&](QObject *w)
{
2017-06-14 14:45:43 -05:00
qDebug() << "window changed:" << qobject_cast<PlasmaWindow *>(w)->appId();
2017-06-10 16:33:46 -05:00
emit windowChanged(qobject_cast<PlasmaWindow *>(w)->internalId());
});
2017-06-10 16:33:46 -05:00
m_windows.push_back(w->internalId());
2017-06-10 16:33:46 -05:00
emit windowAdded(w->internalId());
}
}