Files
qt-creator/src/plugins/scxmleditor/common/structure.cpp

298 lines
10 KiB
C++
Raw Normal View History

/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of Qt Creator.
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3 as published by the Free Software
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
** included in the packaging of this file. Please review the following
** information to ensure the GNU General Public License requirements will
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
**
****************************************************************************/
#include "structure.h"
#include "actionhandler.h"
#include "actionprovider.h"
#include "graphicsscene.h"
#include "sceneutils.h"
#include "scxmleditorconstants.h"
#include "scxmldocument.h"
#include "scxmlnamespace.h"
#include "scxmltagutils.h"
#include "scxmluifactory.h"
#include "structuremodel.h"
#include <QCheckBox>
#include <QKeyEvent>
#include <QLineEdit>
#include <QRegExp>
#include <QRegExpValidator>
#include <QUndoStack>
#include <utils/utilsicons.h>
#include <utils/qtcassert.h>
using namespace ScxmlEditor::PluginInterface;
using namespace ScxmlEditor::Common;
TreeItemDelegate::TreeItemDelegate(QObject *parent)
: QStyledItemDelegate(parent)
{
}
QWidget *TreeItemDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
if (index.isValid()) {
auto edit = new QLineEdit(parent);
edit->setFocusPolicy(Qt::StrongFocus);
QRegExp rx("^(?!xml)[_a-z][a-z0-9-._]*$");
rx.setCaseSensitivity(Qt::CaseInsensitive);
edit->setValidator(new QRegExpValidator(rx, parent));
return edit;
}
return QStyledItemDelegate::createEditor(parent, option, index);
}
StructureSortFilterProxyModel::StructureSortFilterProxyModel(QObject *parent)
: QSortFilterProxyModel(parent)
{
}
void StructureSortFilterProxyModel::setVisibleTags(const QVector<TagType> &visibleTags)
{
m_visibleTags = visibleTags;
if (!m_visibleTags.contains(Scxml))
m_visibleTags << Scxml;
invalidateFilter();
}
void StructureSortFilterProxyModel::setSourceModel(QAbstractItemModel *sourceModel)
{
m_sourceModel = static_cast<StructureModel*>(sourceModel);
QSortFilterProxyModel::setSourceModel(sourceModel);
}
bool StructureSortFilterProxyModel::filterAcceptsRow(int source_row, const QModelIndex &source_parent) const
{
if (m_sourceModel) {
ScxmlTag *tag = m_sourceModel->getItem(source_parent, source_row);
if (tag) {
ScxmlNamespace *ns = tag->document()->scxmlNamespace(tag->prefix());
bool nsBool = (!ns) || ns->isTagVisible(tag->tagName(false));
return m_visibleTags.contains(tag->tagType()) && nsBool;
}
}
return QSortFilterProxyModel::filterAcceptsRow(source_row, source_parent);
}
Structure::Structure(QWidget *parent)
: QFrame(parent)
{
m_ui.setupUi(this);
m_ui.m_checkboxButton->setIcon(Utils::Icons::FILTER.icon());
addCheckbox(tr("Common states"), State);
addCheckbox(tr("Metadata"), Metadata);
addCheckbox(tr("Other tags"), OnEntry);
addCheckbox(tr("Unknown tags"), UnknownTag);
m_ui.m_tagVisibilityFrame->setVisible(false);
connect(m_ui.m_checkboxButton, &QToolButton::toggled, m_ui.m_tagVisibilityFrame, &QFrame::setVisible);
m_model = new StructureModel(this);
m_proxyModel = new StructureSortFilterProxyModel(this);
m_proxyModel->setSourceModel(m_model);
m_proxyModel->setDynamicSortFilter(false);
// Default set of the visible tags
QVector<TagType> visibleTags;
for (int i = 0; i < Finalize; ++i)
visibleTags << (TagType)i;
m_proxyModel->setVisibleTags(visibleTags);
m_ui.m_structureView->setModel(m_proxyModel);
m_ui.m_structureView->setItemDelegate(new TreeItemDelegate(this));
connect(m_ui.m_structureView, &TreeView::pressed, this, &Structure::rowActivated);
connect(m_ui.m_structureView, &TreeView::rightButtonClicked, this, &Structure::showMenu);
connect(m_ui.m_structureView, &TreeView::entered, this, &Structure::rowEntered);
connect(m_model, &StructureModel::selectIndex, this, &Structure::currentTagChanged);
connect(m_model, &StructureModel::childAdded, this, &Structure::childAdded);
}
void Structure::addCheckbox(const QString &name, TagType type)
{
auto box = new QCheckBox;
box->setText(name);
box->setProperty(Constants::C_SCXMLTAG_TAGTYPE, type);
box->setCheckable(true);
box->setChecked(true);
connect(box, &QCheckBox::clicked, this, &Structure::updateCheckBoxes);
m_ui.m_checkboxLayout->addWidget(box);
m_checkboxes << box;
}
void Structure::updateCheckBoxes()
{
QVector<TagType> visibleTags;
foreach (QCheckBox *box, m_checkboxes) {
if (box->isChecked()) {
switch ((TagType)box->property(Constants::C_SCXMLTAG_TAGTYPE).toInt()) {
case State:
visibleTags << Initial << Final << History << State << Parallel << Transition << InitialTransition;
break;
case Metadata:
visibleTags << Metadata << MetadataItem;
break;
case OnEntry:
visibleTags << OnEntry << OnExit << Raise << If << ElseIf << Else
<< Foreach << Log << DataModel << Data << Assign << Donedata
<< Content << Param << Script << Send << Cancel << Invoke << Finalize;
break;
case UnknownTag:
visibleTags << UnknownTag;
break;
default:
break;
}
}
}
m_proxyModel->setVisibleTags(visibleTags);
}
void Structure::setDocument(ScxmlDocument *document)
{
m_currentDocument = document;
m_model->setDocument(document);
m_proxyModel->invalidate();
m_ui.m_structureView->expandAll();
}
void Structure::setGraphicsScene(GraphicsScene *scene)
{
m_scene = scene;
connect(m_ui.m_structureView, &TreeView::mouseExited, m_scene, &GraphicsScene::unhighlightAll);
}
void Structure::rowEntered(const QModelIndex &index)
{
QTC_ASSERT(m_scene, return);
QModelIndex ind = m_proxyModel->mapToSource(index);
auto tag = static_cast<ScxmlTag*>(ind.internalPointer());
if (tag)
m_scene->highlightItems(QVector<ScxmlTag*>() << tag);
else
m_scene->unhighlightAll();
}
void Structure::rowActivated(const QModelIndex &index)
{
if (m_scene)
m_scene->unselectAll();
if (m_currentDocument) {
QModelIndex ind = m_proxyModel->mapToSource(index);
auto tag = static_cast<ScxmlTag*>(ind.internalPointer());
if (tag)
m_currentDocument->setCurrentTag(tag);
}
}
void Structure::currentTagChanged(const QModelIndex &sourceIndex)
{
QModelIndex ind = m_proxyModel->mapFromSource(sourceIndex);
if (ind.isValid())
m_ui.m_structureView->setCurrentIndex(ind);
}
void Structure::childAdded(const QModelIndex &childIndex)
{
m_proxyModel->invalidate();
QModelIndex ind = m_proxyModel->mapFromSource(childIndex);
if (ind.isValid()) {
m_ui.m_structureView->setCurrentIndex(ind);
m_ui.m_structureView->expand(ind.parent());
}
}
void Structure::keyPressEvent(QKeyEvent *e)
{
if (e->key() == Qt::Key_Delete) {
QModelIndex ind = m_proxyModel->mapToSource(m_ui.m_structureView->currentIndex());
auto tag = static_cast<ScxmlTag*>(ind.internalPointer());
if (tag && m_currentDocument) {
m_currentDocument->undoStack()->beginMacro(tr("Remove items"));
m_currentDocument->removeTag(tag);
m_currentDocument->undoStack()->endMacro();
}
}
QFrame::keyPressEvent(e);
}
void Structure::showMenu(const QModelIndex &index, const QPoint &globalPos)
{
if (index.isValid()) {
QModelIndex ind = m_proxyModel->mapToSource(index);
auto tag = static_cast<ScxmlTag*>(ind.internalPointer());
if (tag) {
auto menu = new QMenu;
menu->addAction(tr("Expand All"), m_ui.m_structureView, &TreeView::expandAll);
menu->addAction(tr("Collapse All"), m_ui.m_structureView, &TreeView::collapseAll);
menu->addSeparator();
menu->addAction(m_scene->actionHandler()->action(ActionCopy));
menu->addAction(m_scene->actionHandler()->action(ActionPaste));
menu->addSeparator();
ScxmlUiFactory *uiFactory = m_scene->uiFactory();
if (uiFactory) {
auto actionProvider = static_cast<ActionProvider*>(uiFactory->object(Constants::C_UI_FACTORY_OBJECT_ACTIONPROVIDER));
if (actionProvider) {
actionProvider->initStateMenu(tag, menu);
menu->addSeparator();
}
}
TagUtils::createChildMenu(tag, menu);
QAction *selectedAction = menu->exec(globalPos);
if (selectedAction) {
QVariantMap data = selectedAction->data().toMap();
int actionType = data.value(Constants::C_SCXMLTAG_ACTIONTYPE, -1).toInt();
if (actionType == TagUtils::Remove) {
m_currentDocument->undoStack()->beginMacro(tr("Remove items"));
m_currentDocument->setCurrentTag(tag);
m_currentDocument->removeTag(tag);
m_currentDocument->setCurrentTag(0);
m_currentDocument->undoStack()->endMacro();
} else if (actionType == TagUtils::AddChild) {
tag->document()->undoStack()->beginMacro(tr("Add child"));
ScxmlTag *childTag = SceneUtils::addChild(tag, data, m_scene);
if (childTag && childTag->tagType() <= MetadataItem)
m_ui.m_structureView->edit(m_ui.m_structureView->currentIndex());
tag->document()->undoStack()->endMacro();
}
}
m_proxyModel->invalidate();
menu->deleteLater();
}
}
}