plasma-workspace/appmenu/appmenu.cpp

263 lines
9.6 KiB
C++

/*
SPDX-FileCopyrightText: 2011 Lionel Chauvin <megabigbug@yahoo.fr>
SPDX-FileCopyrightText: 2011, 2012 Cédric Bellegarde <gnumdk@gmail.com>
SPDX-FileCopyrightText: 2016 Kai Uwe Broulik <kde@privat.broulik.de>
SPDX-License-Identifier: MIT
*/
#include <config-X11.h>
#include "appmenu.h"
#include "appmenu_dbus.h"
#include "appmenuadaptor.h"
#include "kdbusimporter.h"
#include "menuimporteradaptor.h"
#include "verticalmenu.h"
#include <QApplication>
#include <QDBusInterface>
#include <QMenu>
#include <private/qwaylanddisplay_p.h>
#include <private/qwaylandinputdevice_p.h>
#include <private/qwaylandwindow_p.h>
#include <KWayland/Client/connection_thread.h>
#include <KWayland/Client/plasmashell.h>
#include <KWayland/Client/registry.h>
#include <KWayland/Client/surface.h>
#include <kpluginfactory.h>
#if HAVE_X11
#include <QX11Info>
#include <xcb/xcb.h>
#endif
static const QByteArray s_x11AppMenuServiceNamePropertyName = QByteArrayLiteral("_KDE_NET_WM_APPMENU_SERVICE_NAME");
static const QByteArray s_x11AppMenuObjectPathPropertyName = QByteArrayLiteral("_KDE_NET_WM_APPMENU_OBJECT_PATH");
K_PLUGIN_FACTORY_WITH_JSON(AppMenuFactory, "appmenu.json", registerPlugin<AppMenuModule>();)
AppMenuModule::AppMenuModule(QObject *parent, const QList<QVariant> &)
: KDEDModule(parent)
, m_appmenuDBus(new AppmenuDBus(this))
{
reconfigure();
m_appmenuDBus->connectToBus();
connect(m_appmenuDBus, &AppmenuDBus::appShowMenu, this, &AppMenuModule::slotShowMenu);
connect(m_appmenuDBus, &AppmenuDBus::reconfigured, this, &AppMenuModule::reconfigure);
// transfer our signals to dbus
connect(this, &AppMenuModule::showRequest, m_appmenuDBus, &AppmenuDBus::showRequest);
connect(this, &AppMenuModule::menuHidden, m_appmenuDBus, &AppmenuDBus::menuHidden);
connect(this, &AppMenuModule::menuShown, m_appmenuDBus, &AppmenuDBus::menuShown);
m_menuViewWatcher = new QDBusServiceWatcher(QStringLiteral("org.kde.kappmenuview"),
QDBusConnection::sessionBus(),
QDBusServiceWatcher::WatchForRegistration | QDBusServiceWatcher::WatchForUnregistration,
this);
auto setupMenuImporter = [this]() {
QDBusConnection::sessionBus().connect({},
{},
QStringLiteral("com.canonical.dbusmenu"),
QStringLiteral("ItemActivationRequested"),
this,
SLOT(itemActivationRequested(int, uint)));
// Setup a menu importer if needed
if (!m_menuImporter) {
m_menuImporter = new MenuImporter(this);
connect(m_menuImporter, &MenuImporter::WindowRegistered, this, &AppMenuModule::slotWindowRegistered);
m_menuImporter->connectToBus();
}
};
connect(m_menuViewWatcher, &QDBusServiceWatcher::serviceRegistered, this, setupMenuImporter);
connect(m_menuViewWatcher, &QDBusServiceWatcher::serviceUnregistered, this, [this](const QString &service) {
Q_UNUSED(service)
QDBusConnection::sessionBus().disconnect({},
{},
QStringLiteral("com.canonical.dbusmenu"),
QStringLiteral("ItemActivationRequested"),
this,
SLOT(itemActivationRequested(int, uint)));
delete m_menuImporter;
m_menuImporter = nullptr;
});
if (QDBusConnection::sessionBus().interface()->isServiceRegistered(QStringLiteral("org.kde.kappmenuview"))) {
setupMenuImporter();
}
#if HAVE_X11
if (!QX11Info::connection()) {
m_xcbConn = xcb_connect(nullptr, nullptr);
}
#endif
if (qGuiApp->platformName() == QLatin1String("wayland")) {
auto connection = KWayland::Client::ConnectionThread::fromApplication();
KWayland::Client::Registry registry;
registry.create(connection);
connect(&registry, &KWayland::Client::Registry::plasmaShellAnnounced, this, [this, &registry](quint32 name, quint32 version) {
m_plasmashell = registry.createPlasmaShell(name, version, this);
});
registry.setup();
connection->roundtrip();
}
}
AppMenuModule::~AppMenuModule()
{
#if HAVE_X11
if (m_xcbConn) {
xcb_disconnect(m_xcbConn);
}
#endif
}
void AppMenuModule::slotWindowRegistered(WId id, const QString &serviceName, const QDBusObjectPath &menuObjectPath)
{
#if HAVE_X11
auto *c = QX11Info::connection();
if (!c) {
c = m_xcbConn;
}
if (c) {
static xcb_atom_t s_serviceNameAtom = XCB_ATOM_NONE;
static xcb_atom_t s_objectPathAtom = XCB_ATOM_NONE;
auto setWindowProperty = [c](WId id, xcb_atom_t &atom, const QByteArray &name, const QByteArray &value) {
if (atom == XCB_ATOM_NONE) {
const xcb_intern_atom_cookie_t cookie = xcb_intern_atom(c, false, name.length(), name.constData());
QScopedPointer<xcb_intern_atom_reply_t, QScopedPointerPodDeleter> reply(xcb_intern_atom_reply(c, cookie, nullptr));
if (reply.isNull()) {
return;
}
atom = reply->atom;
if (atom == XCB_ATOM_NONE) {
return;
}
}
auto cookie = xcb_change_property_checked(c, XCB_PROP_MODE_REPLACE, id, atom, XCB_ATOM_STRING, 8, value.length(), value.constData());
xcb_generic_error_t *error;
if ((error = xcb_request_check(c, cookie))) {
qWarning() << "Got an error";
free(error);
return;
}
};
// TODO only set the property if it doesn't already exist
setWindowProperty(id, s_serviceNameAtom, s_x11AppMenuServiceNamePropertyName, serviceName.toUtf8());
setWindowProperty(id, s_objectPathAtom, s_x11AppMenuObjectPathPropertyName, menuObjectPath.path().toUtf8());
}
#endif
}
void AppMenuModule::slotShowMenu(int x, int y, const QString &serviceName, const QDBusObjectPath &menuObjectPath, int actionId)
{
if (!m_menuImporter) {
return;
}
// If menu visible, hide it
if (m_menu && m_menu.data()->isVisible()) {
m_menu.data()->hide();
return;
}
// dbus call by user (for khotkey shortcut)
if (x == -1 || y == -1) {
// We do not know kwin button position, so tell kwin to show menu
Q_EMIT showRequest(serviceName, menuObjectPath, actionId);
return;
}
auto *importer = new KDBusMenuImporter(serviceName, menuObjectPath.path(), this);
QMetaObject::invokeMethod(importer, "updateMenu", Qt::QueuedConnection);
disconnect(importer, nullptr, this, nullptr); // ensure we don't popup multiple times in case the menu updates again later
connect(importer, &KDBusMenuImporter::menuUpdated, this, [=](QMenu *m) {
QMenu *menu = importer->menu();
if (!menu || menu != m) {
return;
}
m_menu = qobject_cast<VerticalMenu *>(menu);
m_menu.data()->setServiceName(serviceName);
m_menu.data()->setMenuObjectPath(menuObjectPath);
connect(m_menu.data(), &QMenu::aboutToHide, this, [this, importer] {
hideMenu();
importer->deleteLater();
});
if (m_plasmashell) {
connect(m_menu.data(), &QMenu::aboutToShow, this, &AppMenuModule::initMenuWayland, Qt::UniqueConnection);
m_menu.data()->popup(QPoint(x, y));
} else {
m_menu.data()->popup(QPoint(x, y) / qApp->devicePixelRatio());
}
QAction *actiontoActivate = importer->actionForId(actionId);
Q_EMIT menuShown(serviceName, menuObjectPath);
if (actiontoActivate) {
m_menu.data()->setActiveAction(actiontoActivate);
}
});
}
void AppMenuModule::hideMenu()
{
if (m_menu) {
Q_EMIT menuHidden(m_menu.data()->serviceName(), m_menu->menuObjectPath());
}
}
void AppMenuModule::itemActivationRequested(int actionId, uint timeStamp)
{
Q_UNUSED(timeStamp);
Q_EMIT showRequest(message().service(), QDBusObjectPath(message().path()), actionId);
}
// this method is not really used anymore but has to be kept for DBus compatibility
void AppMenuModule::reconfigure()
{
}
void AppMenuModule::initMenuWayland()
{
auto window = m_menu->windowHandle();
if (window && m_plasmashell) {
window->setFlag(Qt::FramelessWindowHint);
window->requestActivate();
auto plasmaSurface = m_plasmashell->createSurface(KWayland::Client::Surface::fromWindow(window), m_menu.data());
plasmaSurface->setPosition(window->position());
plasmaSurface->setSkipSwitcher(true);
plasmaSurface->setSkipTaskbar(true);
m_menu->installEventFilter(this);
}
}
bool AppMenuModule::eventFilter(QObject *object, QEvent *event)
{
// HACK we need an input serial to create popups but Qt only sets them on click
if (object == m_menu && event->type() == QEvent::Enter && m_plasmashell) {
auto waylandWindow = dynamic_cast<QtWaylandClient::QWaylandWindow *>(m_menu->windowHandle()->handle());
if (waylandWindow) {
const auto device = waylandWindow->display()->currentInputDevice();
waylandWindow->display()->setLastInputDevice(device, device->pointer()->mEnterSerial, waylandWindow);
}
}
return KDEDModule::eventFilter(object, event);
}
#include "appmenu.moc"