mirror of
https://github.com/Qortal/Brooklyn.git
synced 2025-02-23 15:45:53 +00:00
293 lines
9.0 KiB
C++
293 lines
9.0 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2016 Kai Uwe Broulik <kde@privat.broulik.de>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
|
*/
|
|
|
|
#include "appmenuapplet.h"
|
|
#include "../plugin/appmenumodel.h"
|
|
|
|
#include <QAction>
|
|
#include <QDBusConnection>
|
|
#include <QDBusConnectionInterface>
|
|
#include <QKeyEvent>
|
|
#include <QMenu>
|
|
#include <QMouseEvent>
|
|
#include <QQuickItem>
|
|
#include <QQuickWindow>
|
|
#include <QScreen>
|
|
#include <QTimer>
|
|
|
|
int AppMenuApplet::s_refs = 0;
|
|
namespace
|
|
{
|
|
QString viewService()
|
|
{
|
|
return QStringLiteral("org.kde.kappmenuview");
|
|
}
|
|
}
|
|
|
|
AppMenuApplet::AppMenuApplet(QObject *parent, const KPluginMetaData &data, const QVariantList &args)
|
|
: Plasma::Applet(parent, data, args)
|
|
{
|
|
++s_refs;
|
|
// if we're the first, register the service
|
|
if (s_refs == 1) {
|
|
QDBusConnection::sessionBus().interface()->registerService(viewService(),
|
|
QDBusConnectionInterface::QueueService,
|
|
QDBusConnectionInterface::DontAllowReplacement);
|
|
}
|
|
/*it registers or unregisters the service when the destroyed value of the applet change,
|
|
and not in the dtor, because:
|
|
when we "delete" an applet, it just hides it for about a minute setting its status
|
|
to destroyed, in order to be able to do a clean undo: if we undo, there will be
|
|
another destroyedchanged and destroyed will be false.
|
|
When this happens, if we are the only appmenu applet existing, the dbus interface
|
|
will have to be registered again*/
|
|
connect(this, &Applet::destroyedChanged, this, [](bool destroyed) {
|
|
if (destroyed) {
|
|
// if we were the last, unregister
|
|
if (--s_refs == 0) {
|
|
QDBusConnection::sessionBus().interface()->unregisterService(viewService());
|
|
}
|
|
} else {
|
|
// if we're the first, register the service
|
|
if (++s_refs == 1) {
|
|
QDBusConnection::sessionBus().interface()->registerService(viewService(),
|
|
QDBusConnectionInterface::QueueService,
|
|
QDBusConnectionInterface::DontAllowReplacement);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
AppMenuApplet::~AppMenuApplet() = default;
|
|
|
|
void AppMenuApplet::init()
|
|
{
|
|
}
|
|
|
|
AppMenuModel *AppMenuApplet::model() const
|
|
{
|
|
return m_model;
|
|
}
|
|
|
|
void AppMenuApplet::setModel(AppMenuModel *model)
|
|
{
|
|
if (m_model != model) {
|
|
m_model = model;
|
|
Q_EMIT modelChanged();
|
|
}
|
|
}
|
|
|
|
int AppMenuApplet::view() const
|
|
{
|
|
return m_viewType;
|
|
}
|
|
|
|
void AppMenuApplet::setView(int type)
|
|
{
|
|
if (m_viewType != type) {
|
|
m_viewType = type;
|
|
Q_EMIT viewChanged();
|
|
}
|
|
}
|
|
|
|
int AppMenuApplet::currentIndex() const
|
|
{
|
|
return m_currentIndex;
|
|
}
|
|
|
|
void AppMenuApplet::setCurrentIndex(int currentIndex)
|
|
{
|
|
if (m_currentIndex != currentIndex) {
|
|
m_currentIndex = currentIndex;
|
|
Q_EMIT currentIndexChanged();
|
|
}
|
|
}
|
|
|
|
QQuickItem *AppMenuApplet::buttonGrid() const
|
|
{
|
|
return m_buttonGrid;
|
|
}
|
|
|
|
void AppMenuApplet::setButtonGrid(QQuickItem *buttonGrid)
|
|
{
|
|
if (m_buttonGrid != buttonGrid) {
|
|
m_buttonGrid = buttonGrid;
|
|
Q_EMIT buttonGridChanged();
|
|
}
|
|
}
|
|
|
|
QMenu *AppMenuApplet::createMenu(int idx) const
|
|
{
|
|
QMenu *menu = nullptr;
|
|
QAction *action = nullptr;
|
|
|
|
if (view() == CompactView) {
|
|
menu = new QMenu();
|
|
for (int i = 0; i < m_model->rowCount(); i++) {
|
|
const QModelIndex index = m_model->index(i, 0);
|
|
const QVariant data = m_model->data(index, AppMenuModel::ActionRole);
|
|
action = (QAction *)data.value<void *>();
|
|
menu->addAction(action);
|
|
}
|
|
menu->setAttribute(Qt::WA_DeleteOnClose);
|
|
} else if (view() == FullView) {
|
|
const QModelIndex index = m_model->index(idx, 0);
|
|
const QVariant data = m_model->data(index, AppMenuModel::ActionRole);
|
|
action = (QAction *)data.value<void *>();
|
|
if (action) {
|
|
menu = action->menu();
|
|
}
|
|
}
|
|
|
|
return menu;
|
|
}
|
|
|
|
void AppMenuApplet::onMenuAboutToHide()
|
|
{
|
|
setCurrentIndex(-1);
|
|
}
|
|
|
|
void AppMenuApplet::trigger(QQuickItem *ctx, int idx)
|
|
{
|
|
if (m_currentIndex == idx) {
|
|
return;
|
|
}
|
|
|
|
if (!ctx || !ctx->window() || !ctx->window()->screen()) {
|
|
return;
|
|
}
|
|
|
|
QMenu *actionMenu = createMenu(idx);
|
|
if (actionMenu) {
|
|
// this is a workaround where Qt will fail to realize a mouse has been released
|
|
// this happens if a window which does not accept focus spawns a new window that takes focus and X grab
|
|
// whilst the mouse is depressed
|
|
// https://bugreports.qt.io/browse/QTBUG-59044
|
|
// this causes the next click to go missing
|
|
|
|
// by releasing manually we avoid that situation
|
|
auto ungrabMouseHack = [ctx]() {
|
|
if (ctx && ctx->window() && ctx->window()->mouseGrabberItem()) {
|
|
// FIXME event forge thing enters press and hold move mode :/
|
|
ctx->window()->mouseGrabberItem()->ungrabMouse();
|
|
}
|
|
};
|
|
|
|
QTimer::singleShot(0, ctx, ungrabMouseHack);
|
|
// end workaround
|
|
|
|
const auto &geo = ctx->window()->screen()->availableVirtualGeometry();
|
|
|
|
QPoint pos = ctx->window()->mapToGlobal(ctx->mapToScene(QPointF()).toPoint());
|
|
if (location() == Plasma::Types::TopEdge) {
|
|
actionMenu->setProperty("_breeze_menu_is_top", true);
|
|
pos.setY(pos.y() + ctx->height());
|
|
}
|
|
|
|
actionMenu->adjustSize();
|
|
|
|
pos = QPoint(qBound(geo.x(), pos.x(), geo.x() + geo.width() - actionMenu->width()),
|
|
qBound(geo.y(), pos.y(), geo.y() + geo.height() - actionMenu->height()));
|
|
|
|
if (view() == FullView) {
|
|
actionMenu->installEventFilter(this);
|
|
}
|
|
|
|
actionMenu->winId(); // create window handle
|
|
actionMenu->windowHandle()->setTransientParent(ctx->window());
|
|
|
|
// hide the old menu only after showing the new one to avoid brief focus flickering on X11.
|
|
// on wayland, you can't have more than one grabbing popup at a time so we show it after
|
|
// the menu has hidden. thankfully, wayland doesn't have this flickering.
|
|
if (!KWindowSystem::isPlatformWayland()) {
|
|
actionMenu->popup(pos);
|
|
}
|
|
|
|
if (view() == FullView) {
|
|
QMenu *oldMenu = m_currentMenu;
|
|
m_currentMenu = actionMenu;
|
|
if (oldMenu && oldMenu != actionMenu) {
|
|
// don't initialize the currentIndex when another menu is already shown
|
|
disconnect(oldMenu, &QMenu::aboutToHide, this, &AppMenuApplet::onMenuAboutToHide);
|
|
oldMenu->hide();
|
|
}
|
|
}
|
|
|
|
if (KWindowSystem::isPlatformWayland()) {
|
|
actionMenu->popup(pos);
|
|
}
|
|
|
|
setCurrentIndex(idx);
|
|
|
|
// FIXME TODO connect only once
|
|
connect(actionMenu, &QMenu::aboutToHide, this, &AppMenuApplet::onMenuAboutToHide, Qt::UniqueConnection);
|
|
} else { // is it just an action without a menu?
|
|
const QVariant data = m_model->index(idx, 0).data(AppMenuModel::ActionRole);
|
|
QAction *action = static_cast<QAction *>(data.value<void *>());
|
|
if (action) {
|
|
Q_ASSERT(!action->menu());
|
|
action->trigger();
|
|
}
|
|
}
|
|
}
|
|
|
|
// FIXME TODO doesn't work on submenu
|
|
bool AppMenuApplet::eventFilter(QObject *watched, QEvent *event)
|
|
{
|
|
auto *menu = qobject_cast<QMenu *>(watched);
|
|
if (!menu) {
|
|
return false;
|
|
}
|
|
|
|
if (event->type() == QEvent::KeyPress) {
|
|
auto *e = static_cast<QKeyEvent *>(event);
|
|
|
|
// TODO right to left languages
|
|
if (e->key() == Qt::Key_Left) {
|
|
int desiredIndex = m_currentIndex - 1;
|
|
Q_EMIT requestActivateIndex(desiredIndex);
|
|
return true;
|
|
} else if (e->key() == Qt::Key_Right) {
|
|
if (menu->activeAction() && menu->activeAction()->menu()) {
|
|
return false;
|
|
}
|
|
|
|
int desiredIndex = m_currentIndex + 1;
|
|
Q_EMIT requestActivateIndex(desiredIndex);
|
|
return true;
|
|
}
|
|
|
|
} else if (event->type() == QEvent::MouseMove) {
|
|
auto *e = static_cast<QMouseEvent *>(event);
|
|
|
|
if (!m_buttonGrid || !m_buttonGrid->window()) {
|
|
return false;
|
|
}
|
|
|
|
// FIXME the panel margin breaks Fitt's law :(
|
|
const QPointF &windowLocalPos = m_buttonGrid->window()->mapFromGlobal(e->globalPos());
|
|
const QPointF &buttonGridLocalPos = m_buttonGrid->mapFromScene(windowLocalPos);
|
|
auto *item = m_buttonGrid->childAt(buttonGridLocalPos.x(), buttonGridLocalPos.y());
|
|
if (!item) {
|
|
return false;
|
|
}
|
|
|
|
bool ok;
|
|
const int buttonIndex = item->property("buttonIndex").toInt(&ok);
|
|
if (!ok) {
|
|
return false;
|
|
}
|
|
|
|
Q_EMIT requestActivateIndex(buttonIndex);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
K_PLUGIN_CLASS_WITH_JSON(AppMenuApplet, "../package/metadata.json")
|
|
|
|
#include "appmenuapplet.moc"
|