Files
qt-creator/src/plugins/coreplugin/actionmanager/actioncontainer.cpp

479 lines
12 KiB
C++
Raw Normal View History

/**************************************************************************
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator
**
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
2008-12-02 12:01:29 +01:00
**
** Contact: Qt Software Information (qt-info@nokia.com)
**
** Commercial Usage
**
** Licensees holding valid Qt Commercial licenses may use this file in
** accordance with the Qt Commercial License Agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Nokia.
**
** GNU Lesser General Public License Usage
**
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** If you are unsure which license is appropriate for your use, please
** contact the sales department at qt-sales@nokia.com.
2008-12-02 12:01:29 +01:00
**
**************************************************************************/
2008-12-02 14:09:21 +01:00
#include "actioncontainer_p.h"
#include "actionmanager_p.h"
#include "command_p.h"
2008-12-02 12:01:29 +01:00
#include "coreconstants.h"
#include "uniqueidmanager.h"
#include <QtCore/QDebug>
#include <QtGui/QAction>
#include <QtGui/QToolBar>
#include <QtGui/QMenuBar>
Q_DECLARE_METATYPE(Core::Internal::MenuActionContainer*)
using namespace Core;
using namespace Core::Internal;
/*!
\class ActionContainer
2008-12-02 12:01:29 +01:00
\mainclass
\brief The ActionContainer class represents a menu or menu bar in Qt Creator.
2008-12-02 12:01:29 +01:00
2008-12-02 12:01:29 +01:00
*/
/*!
\enum ActionContainer::ContainerType
2008-12-02 12:01:29 +01:00
*/
/*!
\enum ActionContainer::EmptyAction
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual ActionContainer::setEmptyAction(EmptyAction ea)
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual int ActionContainer::id() const
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual ContainerType ActionContainer::type() const
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual QMenu *ActionContainer::menu() const
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual QToolBar *ActionContainer::toolBar() const
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual QMenuBar *ActionContainer::menuBar() const
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual QAction *ActionContainer::insertLocation(const QString &group) const
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual void ActionContainer::appendGroup(const QString &group, bool global)
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual void ActionContainer::addAction(Core::Command *action, const QString &group)
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual void ActionContainer::addMenu(Core::ActionContainer *menu, const QString &group)
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual bool ActionContainer::update()
2008-12-02 12:01:29 +01:00
*/
/*!
\fn virtual ActionContainer::~ActionContainer()
2008-12-02 12:01:29 +01:00
*/
// ---------- ActionContainerPrivate ------------
2008-12-02 12:01:29 +01:00
/*!
\class Core::Internal::ActionContainerPrivate
\internal
2008-12-02 12:01:29 +01:00
*/
ActionContainerPrivate::ActionContainerPrivate(int id)
: m_data(CS_None), m_id(id)
2008-12-02 12:01:29 +01:00
{
}
void ActionContainerPrivate::setEmptyAction(EmptyAction ea)
2008-12-02 12:01:29 +01:00
{
m_data = ((m_data & ~EA_Mask) | ea);
}
bool ActionContainerPrivate::hasEmptyAction(EmptyAction ea) const
2008-12-02 12:01:29 +01:00
{
return (m_data & EA_Mask) == ea;
}
void ActionContainerPrivate::setState(ContainerState state)
2008-12-02 12:01:29 +01:00
{
m_data |= state;
}
bool ActionContainerPrivate::hasState(ContainerState state) const
2008-12-02 12:01:29 +01:00
{
return (m_data & state);
}
void ActionContainerPrivate::appendGroup(const QString &group)
2008-12-02 12:01:29 +01:00
{
int gid = UniqueIDManager::instance()->uniqueIdentifier(group);
2008-12-02 12:01:29 +01:00
m_groups << gid;
}
QAction *ActionContainerPrivate::insertLocation(const QString &group) const
2008-12-02 12:01:29 +01:00
{
int grpid = UniqueIDManager::instance()->uniqueIdentifier(group);
2008-12-02 12:01:29 +01:00
int prevKey = 0;
int pos = ((grpid << 16) | 0xFFFF);
return beforeAction(pos, &prevKey);
}
void ActionContainerPrivate::addAction(Command *action, const QString &group)
2008-12-02 12:01:29 +01:00
{
if (!canAddAction(action))
return;
ActionManagerPrivate *am = ActionManagerPrivate::instance();
2008-12-02 12:01:29 +01:00
Action *a = static_cast<Action *>(action);
if (a->stateFlags() & CommandPrivate::CS_PreLocation) {
2008-12-02 12:01:29 +01:00
QList<CommandLocation> locs = a->locations();
for (int i=0; i<locs.size(); ++i) {
if (ActionContainer *aci = am->actionContainer(locs.at(i).m_container)) {
ActionContainerPrivate *ac = static_cast<ActionContainerPrivate *>(aci);
2008-12-02 12:01:29 +01:00
ac->addAction(action, locs.at(i).m_position, false);
}
}
a->setStateFlags(a->stateFlags() | CommandPrivate::CS_Initialized);
2008-12-02 12:01:29 +01:00
} else {
UniqueIDManager *idmanager = UniqueIDManager::instance();
2008-12-02 12:01:29 +01:00
int grpid = idmanager->uniqueIdentifier(Constants::G_DEFAULT_TWO);
if (!group.isEmpty())
grpid = idmanager->uniqueIdentifier(group);
if (!m_groups.contains(grpid) && !am->defaultGroups().contains(grpid))
qWarning() << "*** addAction(): Unknown group: " << group;
int pos = ((grpid << 16) | 0xFFFF);
addAction(action, pos, true);
}
}
void ActionContainerPrivate::addMenu(ActionContainer *menu, const QString &group)
2008-12-02 12:01:29 +01:00
{
ActionContainerPrivate *container = static_cast<ActionContainerPrivate *>(menu);
if (!container->canBeAddedToMenu())
2008-12-02 12:01:29 +01:00
return;
ActionManagerPrivate *am = ActionManagerPrivate::instance();
2008-12-02 12:01:29 +01:00
MenuActionContainer *mc = static_cast<MenuActionContainer *>(menu);
if (mc->hasState(ActionContainerPrivate::CS_PreLocation)) {
2008-12-02 12:01:29 +01:00
CommandLocation loc = mc->location();
if (ActionContainer *aci = am->actionContainer(loc.m_container)) {
ActionContainerPrivate *ac = static_cast<ActionContainerPrivate *>(aci);
2008-12-02 12:01:29 +01:00
ac->addMenu(menu, loc.m_position, false);
}
mc->setState(ActionContainerPrivate::CS_Initialized);
2008-12-02 12:01:29 +01:00
} else {
UniqueIDManager *idmanager = UniqueIDManager::instance();
2008-12-02 12:01:29 +01:00
int grpid = idmanager->uniqueIdentifier(Constants::G_DEFAULT_TWO);
if (!group.isEmpty())
grpid = idmanager->uniqueIdentifier(group);
if (!m_groups.contains(grpid) && !am->defaultGroups().contains(grpid))
qWarning() << "*** addMenu(): Unknown group: " << group;
int pos = ((grpid << 16) | 0xFFFF);
addMenu(menu, pos, true);
}
}
int ActionContainerPrivate::id() const
2008-12-02 12:01:29 +01:00
{
return m_id;
}
QMenu *ActionContainerPrivate::menu() const
2008-12-02 12:01:29 +01:00
{
return 0;
}
QMenuBar *ActionContainerPrivate::menuBar() const
2008-12-02 12:01:29 +01:00
{
return 0;
}
bool ActionContainerPrivate::canAddAction(Command *action) const
2008-12-02 12:01:29 +01:00
{
if (action->type() != Command::CT_OverridableAction)
2008-12-02 12:01:29 +01:00
return false;
CommandPrivate *cmd = static_cast<CommandPrivate *>(action);
if (cmd->stateFlags() & CommandPrivate::CS_Initialized)
2008-12-02 12:01:29 +01:00
return false;
return true;
}
void ActionContainerPrivate::addAction(Command *action, int pos, bool setpos)
2008-12-02 12:01:29 +01:00
{
Action *a = static_cast<Action *>(action);
int prevKey = 0;
QAction *ba = beforeAction(pos, &prevKey);
if (setpos) {
pos = calcPosition(pos, prevKey);
CommandLocation loc;
loc.m_container = m_id;
loc.m_position = pos;
QList<CommandLocation> locs = a->locations();
locs.append(loc);
a->setLocations(locs);
}
m_commands.append(action);
m_posmap.insert(pos, action->id());
insertAction(ba, a->action());
}
void ActionContainerPrivate::addMenu(ActionContainer *menu, int pos, bool setpos)
2008-12-02 12:01:29 +01:00
{
MenuActionContainer *mc = static_cast<MenuActionContainer *>(menu);
int prevKey = 0;
QAction *ba = beforeAction(pos, &prevKey);
if (setpos) {
pos = calcPosition(pos, prevKey);
CommandLocation loc;
loc.m_container = m_id;
loc.m_position = pos;
mc->setLocation(loc);
}
m_subContainers.append(menu);
m_posmap.insert(pos, menu->id());
insertMenu(ba, mc->menu());
}
QAction *ActionContainerPrivate::beforeAction(int pos, int *prevKey) const
2008-12-02 12:01:29 +01:00
{
ActionManagerPrivate *am = ActionManagerPrivate::instance();
2008-12-02 12:01:29 +01:00
int baId = -1;
(*prevKey) = -1;
QMap<int, int>::const_iterator i = m_posmap.constBegin();
while (i != m_posmap.constEnd()) {
if (i.key() > pos) {
baId = i.value();
break;
}
(*prevKey) = i.key();
++i;
}
if (baId == -1)
return 0;
if (Command *cmd = am->command(baId))
2008-12-02 12:01:29 +01:00
return cmd->action();
if (ActionContainer *container = am->actionContainer(baId))
2008-12-02 12:01:29 +01:00
if (QMenu *menu = container->menu())
return menu->menuAction();
return 0;
}
int ActionContainerPrivate::calcPosition(int pos, int prevKey) const
2008-12-02 12:01:29 +01:00
{
int grp = (pos & 0xFFFF0000);
if (prevKey == -1)
return grp;
int prevgrp = (prevKey & 0xFFFF0000);
if (grp != prevgrp)
return grp;
return grp + (prevKey & 0xFFFF) + 10;
}
// ---------- MenuActionContainer ------------
/*!
\class Core::Internal::MenuActionContainer
\internal
2008-12-02 12:01:29 +01:00
*/
MenuActionContainer::MenuActionContainer(int id)
: ActionContainerPrivate(id), m_menu(0)
2008-12-02 12:01:29 +01:00
{
setEmptyAction(EA_Disable);
}
void MenuActionContainer::setMenu(QMenu *menu)
{
m_menu = menu;
QVariant v;
qVariantSetValue<MenuActionContainer*>(v, this);
m_menu->menuAction()->setData(v);
}
QMenu *MenuActionContainer::menu() const
{
return m_menu;
}
void MenuActionContainer::insertAction(QAction *before, QAction *action)
{
m_menu->insertAction(before, action);
}
void MenuActionContainer::insertMenu(QAction *before, QMenu *menu)
{
m_menu->insertMenu(before, menu);
}
void MenuActionContainer::setLocation(const CommandLocation &location)
{
m_location = location;
}
CommandLocation MenuActionContainer::location() const
{
return m_location;
}
bool MenuActionContainer::update()
{
if (hasEmptyAction(EA_None))
return true;
bool hasitems = false;
foreach (ActionContainer *container, subContainers()) {
2008-12-02 12:01:29 +01:00
if (container == this) {
qWarning() << Q_FUNC_INFO << "container" << (this->menu() ? this->menu()->title() : "") << "contains itself as subcontainer";
continue;
}
if (container->update()) {
hasitems = true;
break;
}
}
if (!hasitems) {
foreach (Command *command, commands()) {
2008-12-02 12:01:29 +01:00
if (command->isActive()) {
hasitems = true;
break;
}
}
}
if (hasEmptyAction(EA_Hide))
m_menu->setVisible(hasitems);
else if (hasEmptyAction(EA_Disable))
m_menu->setEnabled(hasitems);
return hasitems;
}
bool MenuActionContainer::canBeAddedToMenu() const
2008-12-02 12:01:29 +01:00
{
if (hasState(ActionContainerPrivate::CS_Initialized))
return false;
2008-12-02 12:01:29 +01:00
return true;
2008-12-02 12:01:29 +01:00
}
// ---------- MenuBarActionContainer ------------
/*!
\class Core::Internal::MenuBarActionContainer
\internal
2008-12-02 12:01:29 +01:00
*/
MenuBarActionContainer::MenuBarActionContainer(int id)
: ActionContainerPrivate(id), m_menuBar(0)
2008-12-02 12:01:29 +01:00
{
setEmptyAction(EA_None);
}
void MenuBarActionContainer::setMenuBar(QMenuBar *menuBar)
{
m_menuBar = menuBar;
}
QMenuBar *MenuBarActionContainer::menuBar() const
{
return m_menuBar;
}
void MenuBarActionContainer::insertAction(QAction *before, QAction *action)
{
m_menuBar->insertAction(before, action);
}
void MenuBarActionContainer::insertMenu(QAction *before, QMenu *menu)
{
m_menuBar->insertMenu(before, menu);
}
bool MenuBarActionContainer::update()
{
if (hasEmptyAction(EA_None))
return true;
bool hasitems = false;
QList<QAction *> actions = m_menuBar->actions();
for (int i=0; i<actions.size(); ++i) {
if (actions.at(i)->isVisible()) {
hasitems = true;
break;
}
}
if (hasEmptyAction(EA_Hide))
m_menuBar->setVisible(hasitems);
else if (hasEmptyAction(EA_Disable))
m_menuBar->setEnabled(hasitems);
return hasitems;
}
bool MenuBarActionContainer::canBeAddedToMenu() const
{
return false;
}