a0fb4356dc
Actually quite a bit more lightweight than its QGV predecessor.
195 lines
5.2 KiB
C++
195 lines
5.2 KiB
C++
/***************************************************************************
|
|
* Copyright 2011 Viranch Mehta <viranch.mehta@gmail.com> *
|
|
* *
|
|
* 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, write to the *
|
|
* Free Software Foundation, Inc., *
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA . *
|
|
***************************************************************************/
|
|
|
|
#include "qmenu.h"
|
|
|
|
#include <QApplication>
|
|
#include <QDebug>
|
|
#include <QDesktopWidget>
|
|
#include <QQuickWindow>
|
|
#include <QQuickItem>
|
|
|
|
#include "plasmacomponentsplugin.h"
|
|
QMenuProxy::QMenuProxy (QObject *parent)
|
|
: QObject(parent),
|
|
m_status(DialogStatus::Closed)
|
|
{
|
|
m_menu = new QMenu(0);
|
|
connect(m_menu, SIGNAL(triggered(QAction *)),
|
|
this, SLOT(itemTriggered(QAction *)));
|
|
}
|
|
|
|
QMenuProxy::~QMenuProxy()
|
|
{
|
|
delete m_menu;
|
|
}
|
|
|
|
QQmlListProperty<QMenuItem> QMenuProxy::content()
|
|
{
|
|
return QQmlListProperty<QMenuItem>(this, m_items);
|
|
}
|
|
|
|
int QMenuProxy::actionCount() const
|
|
{
|
|
return m_items.count();
|
|
}
|
|
|
|
QMenuItem *QMenuProxy::action(int index) const
|
|
{
|
|
return m_items.at(index);
|
|
}
|
|
|
|
DialogStatus::Status QMenuProxy::status() const
|
|
{
|
|
return m_status;
|
|
}
|
|
|
|
QObject *QMenuProxy::visualParent() const
|
|
{
|
|
return m_visualParent.data();
|
|
}
|
|
|
|
void QMenuProxy::setVisualParent(QObject *parent)
|
|
{
|
|
if (m_visualParent.data() == parent) {
|
|
return;
|
|
}
|
|
|
|
//if the old parent was a QAction, disconnect the menu from it
|
|
QAction *action = qobject_cast<QAction *>(m_visualParent.data());
|
|
if (action) {
|
|
action->setMenu(0);
|
|
m_menu->clear();
|
|
}
|
|
//if parent is a QAction, become a submenu
|
|
action = qobject_cast<QAction *>(parent);
|
|
if (action) {
|
|
action->setMenu(m_menu);
|
|
m_menu->clear();
|
|
foreach(QMenuItem* item, m_items) {
|
|
m_menu->addAction(item->action());
|
|
}
|
|
m_menu->updateGeometry();
|
|
}
|
|
|
|
m_visualParent = parent;
|
|
emit visualParentChanged();
|
|
}
|
|
|
|
bool QMenuProxy::event(QEvent *event)
|
|
{
|
|
switch (event->type()) {
|
|
case QEvent::ChildAdded: {
|
|
QChildEvent *ce = static_cast<QChildEvent *>(event);
|
|
QMenuItem *mi = qobject_cast<QMenuItem *>(ce->child());
|
|
//FIXME: linear complexity here
|
|
if (mi && !m_items.contains(mi)) {
|
|
m_menu->addAction(mi->action());
|
|
m_items << mi;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case QEvent::ChildRemoved: {
|
|
QChildEvent *ce = static_cast<QChildEvent *>(event);
|
|
QMenuItem *mi = qobject_cast<QMenuItem *>(ce->child());
|
|
|
|
//FIXME: linear complexity here
|
|
if (mi) {
|
|
m_menu->removeAction(mi->action());
|
|
m_items.removeAll(mi);
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return QObject::event(event);
|
|
}
|
|
|
|
void QMenuProxy::clearMenuItems()
|
|
{
|
|
qDeleteAll(m_items);
|
|
m_items.clear();
|
|
}
|
|
|
|
void QMenuProxy::addMenuItem(const QString &text)
|
|
{
|
|
QMenuItem *item = new QMenuItem();
|
|
item->setText(text);
|
|
m_menu->addAction(item->action());
|
|
m_items << item;
|
|
}
|
|
|
|
void QMenuProxy::addMenuItem(QMenuItem *item)
|
|
{
|
|
m_menu->addAction(item->action());
|
|
m_items << item;
|
|
}
|
|
|
|
void QMenuProxy::itemTriggered(QAction *action)
|
|
{
|
|
QMenuItem *item = qobject_cast<QMenuItem *>(action);
|
|
if (item) {
|
|
emit triggered(item);
|
|
int index = m_items.indexOf(item);
|
|
if (index > -1) {
|
|
emit triggeredIndex(index);
|
|
}
|
|
}
|
|
}
|
|
|
|
void QMenuProxy::open(int x, int y)
|
|
{
|
|
qDebug() << "opening menu at " << x << y;
|
|
m_menu->clear();
|
|
foreach(QMenuItem* item, m_items) {
|
|
qDebug() <<"Adding action: " << item->text();
|
|
m_menu->addAction(item->action());
|
|
}
|
|
|
|
QQuickItem *parentItem;
|
|
if (m_visualParent) {
|
|
parentItem = qobject_cast<QQuickItem *>(m_visualParent.data());
|
|
} else {
|
|
parentItem = qobject_cast<QQuickItem *>(parent());
|
|
}
|
|
QPointF pos = parentItem->mapToScene(QPointF(x, y));
|
|
if (parentItem->window() && parentItem->window()->screen()) {
|
|
pos = parentItem->window()->mapToGlobal(pos.toPoint());
|
|
}
|
|
|
|
m_menu->popup(pos.toPoint());
|
|
m_status = DialogStatus::Open;
|
|
emit statusChanged();
|
|
}
|
|
|
|
|
|
void QMenuProxy::close()
|
|
{
|
|
m_menu->hide();
|
|
m_status = DialogStatus::Closed;
|
|
emit statusChanged();
|
|
}
|
|
|
|
#include "qmenu.moc"
|
|
|