shift-shell/containments/taskpanel/taskpanel.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

311 lines
10 KiB
C++
Raw Normal View History

2021-03-01 20:03:25 +00:00
/*
* SPDX-FileCopyrightText: 2015 Marco Martin <mart@kde.org>
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "taskpanel.h"
#include <QDebug>
#include <QQuickItem>
#include <QQuickWindow>
#include <QtQml>
#include <KWayland/Client/connection_thread.h>
#include <KWayland/Client/output.h>
#include <KWayland/Client/plasmashell.h>
#include <KWayland/Client/plasmawindowmanagement.h>
#include <KWayland/Client/registry.h>
#include <KWayland/Client/surface.h>
#include <Plasma/Package>
#include <virtualkeyboardinterface.h>
static const QString s_kwinService = QStringLiteral("org.kde.KWin");
2020-03-20 00:08:59 +00:00
constexpr int ACTIVE_WINDOW_UPDATE_INVERVAL = 250;
class OutputsModel : public QAbstractListModel
{
public:
enum Roles {
Model = Qt::DisplayRole,
Geometry = Qt::UserRole,
Position,
Output,
};
OutputsModel(QObject *parent)
: QAbstractListModel(parent)
{
}
QHash<int, QByteArray> roleNames() const override
{
return {
{Model, "modelName"},
{Geometry, "geometry"},
{Position, "position"},
{Output, "output"},
};
}
void createOutput(wl_output *output)
{
auto newOutput = new KWayland::Client::Output(this);
connect(newOutput, &KWayland::Client::Output::removed, this, [this, newOutput] {
auto i = m_outputs.indexOf(newOutput);
Q_ASSERT(i >= 0);
beginRemoveRows({}, i, i);
m_outputs.removeAt(i);
endRemoveRows();
});
newOutput->setup(output);
beginInsertRows({}, m_outputs.count(), m_outputs.count());
m_outputs.append(newOutput);
endInsertRows();
}
int rowCount(const QModelIndex &parent) const override
{
return parent.isValid() ? 0 : m_outputs.count();
}
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override
{
if (index.row() >= m_outputs.count()) {
return {};
}
auto o = m_outputs[index.row()];
switch (role) {
case Model:
return o->model();
case Geometry:
return o->geometry();
case Position:
return o->globalPosition();
case Output:
return QVariant::fromValue<QObject *>(o);
}
return {};
}
private:
QVector<KWayland::Client::Output *> m_outputs;
};
// helper class to expose the NOTIFY in the properties
class KwinVirtualKeyboardInterface : public OrgKdeKwinVirtualKeyboardInterface
{
Q_OBJECT
Q_PROPERTY(bool active READ active WRITE setActive NOTIFY activeChanged)
Q_PROPERTY(bool enabled READ enabled WRITE setEnabled NOTIFY enabledChanged)
Q_PROPERTY(bool visible READ visible NOTIFY visibleChanged)
public:
KwinVirtualKeyboardInterface()
: OrgKdeKwinVirtualKeyboardInterface(QStringLiteral("org.kde.KWin"), QStringLiteral("/VirtualKeyboard"), QDBusConnection::sessionBus())
{
}
};
TaskPanel::TaskPanel(QObject *parent, const QVariantList &args)
: Plasma::Containment(parent, args)
, m_showingDesktop(false)
, m_windowManagement(nullptr)
, m_outputsModel(new OutputsModel(this))
{
setHasConfigurationInterface(true);
m_activeTimer = new QTimer(this);
m_activeTimer->setSingleShot(true);
2020-03-20 00:08:59 +00:00
m_activeTimer->setInterval(ACTIVE_WINDOW_UPDATE_INVERVAL);
connect(m_activeTimer, &QTimer::timeout, this, &TaskPanel::updateActiveWindow);
initWayland();
qmlRegisterUncreatableType<KWayland::Client::Output>("org.kde.plasma.phone.taskpanel", 1, 0, "Output", "nope");
qmlRegisterUncreatableType<OutputsModel>("org.kde.plasma.phone.taskpanel", 1, 0, "OutputsModel", "nope");
qmlRegisterSingletonType<OrgKdeKwinVirtualKeyboardInterface>("org.kde.plasma.phone.taskpanel",
1,
0,
"KWinVirtualKeyboard",
[](QQmlEngine *, QJSEngine *) -> QObject * {
return new KwinVirtualKeyboardInterface;
});
connect(this, &Plasma::Containment::locationChanged, this, &TaskPanel::locationChanged);
connect(this, &Plasma::Containment::locationChanged, this, [this] {
auto l = location();
setFormFactor(l == Plasma::Types::LeftEdge || l == Plasma::Types::RightEdge ? Plasma::Types::Vertical : Plasma::Types::Horizontal);
});
}
2020-03-20 00:08:59 +00:00
TaskPanel::~TaskPanel() = default;
void TaskPanel::requestShowingDesktop(bool showingDesktop)
{
if (!m_windowManagement) {
return;
}
m_windowManagement->setShowingDesktop(showingDesktop);
}
void TaskPanel::initWayland()
{
if (!QGuiApplication::platformName().startsWith(QLatin1String("wayland"), Qt::CaseInsensitive)) {
return;
}
using namespace KWayland::Client;
ConnectionThread *connection = ConnectionThread::fromApplication(this);
2020-07-23 11:47:40 +00:00
if (!connection) {
return;
}
2019-05-03 14:19:31 +00:00
auto *registry = new Registry(this);
registry->create(connection);
connect(registry, &Registry::outputAnnounced, m_outputsModel, [this, registry](quint32 name, quint32 version) {
m_outputsModel->createOutput(registry->bindOutput(name, version));
});
connect(registry, &Registry::plasmaWindowManagementAnnounced, this, [this, registry](quint32 name, quint32 version) {
m_windowManagement = registry->createPlasmaWindowManagement(name, version, this);
qRegisterMetaType<QVector<int>>("QVector<int>");
connect(m_windowManagement, &PlasmaWindowManagement::showingDesktopChanged, this, [this](bool showing) {
if (showing == m_showingDesktop) {
return;
}
m_showingDesktop = showing;
emit showingDesktopChanged(m_showingDesktop);
});
// FIXME
// connect(m_windowManagement, &PlasmaWindowManagement::activeWindowChanged, this, &TaskPanel::updateActiveWindow, Qt::QueuedConnection);
connect(m_windowManagement, &KWayland::Client::PlasmaWindowManagement::activeWindowChanged, m_activeTimer, qOverload<>(&QTimer::start));
2020-07-30 13:29:57 +00:00
m_activeTimer->start();
});
connect(registry, &Registry::plasmaShellAnnounced, this, [this, registry](quint32 name, quint32 version) {
m_shellInterface = registry->createPlasmaShell(name, version, this);
if (!m_panel) {
return;
}
Surface *s = Surface::fromWindow(m_panel);
if (!s) {
return;
}
m_shellSurface = m_shellInterface->createSurface(s, this);
m_shellSurface->setSkipTaskbar(true);
});
registry->setup();
connection->roundtrip();
}
QWindow *TaskPanel::panel()
{
return m_panel;
}
void TaskPanel::setPanel(QWindow *panel)
{
if (panel == m_panel) {
return;
}
2017-09-14 11:01:07 +00:00
if (m_panel) {
disconnect(m_panel, &QWindow::visibilityChanged, this, &TaskPanel::updatePanelVisibility);
}
m_panel = panel;
2017-09-14 11:01:07 +00:00
connect(m_panel, &QWindow::visibilityChanged, this, &TaskPanel::updatePanelVisibility, Qt::QueuedConnection);
emit panelChanged();
2017-09-14 11:01:07 +00:00
updatePanelVisibility();
}
void TaskPanel::updatePanelVisibility()
{
using namespace KWayland::Client;
if (!m_panel->isVisible()) {
return;
}
Surface *s = Surface::fromWindow(m_panel);
if (!s) {
return;
}
m_shellSurface = m_shellInterface->createSurface(s, this);
if (m_shellSurface) {
m_shellSurface->setSkipTaskbar(true);
}
}
void TaskPanel::updateActiveWindow()
{
if (!m_windowManagement || m_activeWindow == m_windowManagement->activeWindow()) {
return;
}
if (m_activeWindow) {
disconnect(m_activeWindow.data(), &KWayland::Client::PlasmaWindow::closeableChanged, this, &TaskPanel::hasCloseableActiveWindowChanged);
disconnect(m_activeWindow.data(), &KWayland::Client::PlasmaWindow::unmapped, this, &TaskPanel::forgetActiveWindow);
}
m_activeWindow = m_windowManagement->activeWindow();
if (m_activeWindow) {
connect(m_activeWindow.data(), &KWayland::Client::PlasmaWindow::closeableChanged, this, &TaskPanel::hasCloseableActiveWindowChanged);
connect(m_activeWindow.data(), &KWayland::Client::PlasmaWindow::unmapped, this, &TaskPanel::forgetActiveWindow);
}
2020-07-23 11:47:40 +00:00
bool newAllMinimized = true;
for (auto *w : m_windowManagement->windows()) {
if (!w->isMinimized() && !w->skipTaskbar() && !w->isFullscreen() /*&& w->appId() != QStringLiteral("org.kde.plasmashell")*/) {
newAllMinimized = false;
break;
}
}
if (newAllMinimized != m_allMinimized) {
m_allMinimized = newAllMinimized;
emit allMinimizedChanged();
}
// TODO: connect to closeableChanged, not needed right now as KWin doesn't provide this changeable
emit hasCloseableActiveWindowChanged();
}
bool TaskPanel::hasCloseableActiveWindow() const
{
return m_activeWindow && m_activeWindow->isCloseable() /*&& !m_activeWindow->isMinimized()*/;
}
void TaskPanel::forgetActiveWindow()
{
if (m_activeWindow) {
disconnect(m_activeWindow.data(), &KWayland::Client::PlasmaWindow::closeableChanged, this, &TaskPanel::hasCloseableActiveWindowChanged);
disconnect(m_activeWindow.data(), &KWayland::Client::PlasmaWindow::unmapped, this, &TaskPanel::forgetActiveWindow);
}
m_activeWindow.clear();
emit hasCloseableActiveWindowChanged();
}
void TaskPanel::closeActiveWindow()
{
if (m_activeWindow) {
m_activeWindow->requestClose();
}
}
void TaskPanel::sendWindowToOutput(const QString &uuid, KWayland::Client::Output *output)
{
const auto windows = m_windowManagement->windows();
for (auto w : windows) {
if (w->uuid() == uuid) {
w->sendToOutput(output);
}
}
}
QAbstractItemModel *TaskPanel::outputs() const
{
return m_outputsModel;
}
K_PLUGIN_CLASS_WITH_JSON(TaskPanel, "metadata.json")
#include "taskpanel.moc"