1
0
mirror of https://bitbucket.org/librepilot/librepilot.git synced 2025-01-25 10:52:11 +01:00
LibrePilot/ground/openpilotgcs/src/plugins/pathactioneditor/pathactioneditortreemodel.cpp

452 lines
14 KiB
C++
Raw Normal View History

2013-04-05 23:46:56 +03:00
/**
******************************************************************************
*
* @file pathactioneditortreemodel.cpp
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
* @addtogroup GCSPlugins GCS Plugins
* @{
* @addtogroup UAVObjectBrowserPlugin UAVObject Browser Plugin
* @{
* @brief The UAVObject Browser gadget plugin
*****************************************************************************/
/*
* 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 3 of the License, or
2013-04-05 23:46:56 +03:00
* (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
2013-04-05 23:46:56 +03:00
* 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.,
2013-04-05 23:46:56 +03:00
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "pathactioneditortreemodel.h"
#include "pathactioneditorgadget.h"
#include "fieldtreeitem.h"
#include "uavobjectmanager.h"
#include "uavdataobject.h"
#include "uavmetaobject.h"
#include "uavobjectfield.h"
#include "extensionsystem/pluginmanager.h"
#include <QColor>
// #include <QIcon>
2013-04-05 23:46:56 +03:00
#include <QMessageBox>
#include <QtCore/QSignalMapper>
#include <QtCore/QDebug>
#include "waypoint.h"
#include "waypointactive.h"
#include "pathaction.h"
PathActionEditorTreeModel::PathActionEditorTreeModel(QObject *parent) :
QAbstractItemModel(parent),
m_recentlyUpdatedColor(QColor(255, 230, 230)),
m_manuallyChangedColor(QColor(230, 230, 255))
2013-04-05 23:46:56 +03:00
{
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
2013-04-05 23:46:56 +03:00
m_objManager = pm->getObject<UAVObjectManager>();
connect(m_objManager, SIGNAL(newInstance(UAVObject *)), this, SLOT(newInstance(UAVObject *)));
connect(m_objManager->getObject("WaypointActive"), SIGNAL(objectUpdated(UAVObject *)), this, SLOT(objUpdated(UAVObject *)));
connect(m_objManager->getObject("PathAction"), SIGNAL(objectUpdated(UAVObject *)), this, SLOT(objUpdated(UAVObject *)));
connect(m_objManager->getObject("Waypoint"), SIGNAL(objectUpdated(UAVObject *)), this, SLOT(objUpdated(UAVObject *)));
2013-04-05 23:46:56 +03:00
setupModelData();
}
PathActionEditorTreeModel::~PathActionEditorTreeModel()
{
delete m_rootItem;
}
void PathActionEditorTreeModel::setupModelData()
{
m_actions = new QStringList();
updateActions();
// root
QList<QVariant> rootData;
rootData << tr("Property") << tr("Value") << tr("Unit");
m_rootItem = new TreeItem(rootData);
m_pathactionsTree = new TopTreeItem(tr("PathActions"), m_rootItem);
m_rootItem->appendChild(m_pathactionsTree);
m_waypointsTree = new TopTreeItem(tr("Waypoints"), m_rootItem);
2013-04-05 23:46:56 +03:00
m_rootItem->appendChild(m_waypointsTree);
connect(m_rootItem, SIGNAL(updateHighlight(TreeItem *)), this, SLOT(updateHighlight(TreeItem *)));
connect(m_pathactionsTree, SIGNAL(updateHighlight(TreeItem *)), this, SLOT(updateHighlight(TreeItem *)));
connect(m_waypointsTree, SIGNAL(updateHighlight(TreeItem *)), this, SLOT(updateHighlight(TreeItem *)));
2013-04-05 23:46:56 +03:00
{
QList<UAVObject *> list = m_objManager->getObjectInstances("PathAction");
foreach(UAVObject * obj, list) {
addInstance(obj, m_pathactionsTree);
2013-04-05 23:46:56 +03:00
}
}
{
QList<UAVObject *> list = m_objManager->getObjectInstances("Waypoint");
foreach(UAVObject * obj, list) {
addInstance(obj, m_waypointsTree);
2013-04-05 23:46:56 +03:00
}
}
}
void PathActionEditorTreeModel::updateActions()
{
m_actions->clear();
QList<UAVObject *> list = m_objManager->getObjectInstances("PathAction");
foreach(UAVObject * obj, list) {
QString title;
title.append((QVariant(obj->getInstID()).toString()));
title.append(" ");
title.append((obj->getField("Mode")->getValue().toString()));
title.append(" ");
title.append((obj->getField("Command")->getValue().toString()));
title.append(":");
title.append((obj->getField("EndCondition")->getValue().toString()));
title.append(" ");
m_actions->append(title);
}
2013-04-05 23:46:56 +03:00
}
void PathActionEditorTreeModel::addInstance(UAVObject *obj, TreeItem *parent)
{
connect(obj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(highlightUpdatedObject(UAVObject *)));
2013-04-05 23:46:56 +03:00
TreeItem *item;
QString name = QString::number(obj->getInstID());
item = new InstanceTreeItem(obj, name);
connect(item, SIGNAL(updateHighlight(TreeItem *)), this, SLOT(updateHighlight(TreeItem *)));
2013-04-05 23:46:56 +03:00
parent->appendChild(item);
foreach(UAVObjectField * field, obj->getFields()) {
2013-04-05 23:46:56 +03:00
if (field->getNumElements() > 1) {
addArrayField(field, item);
} else {
addSingleField(0, field, item);
}
}
}
void PathActionEditorTreeModel::addArrayField(UAVObjectField *field, TreeItem *parent)
{
TreeItem *item = new ArrayFieldTreeItem(field->getName());
connect(item, SIGNAL(updateHighlight(TreeItem *)), this, SLOT(updateHighlight(TreeItem *)));
2013-04-05 23:46:56 +03:00
for (uint i = 0; i < field->getNumElements(); ++i) {
addSingleField(i, field, item);
}
parent->appendChild(item);
}
void PathActionEditorTreeModel::addSingleField(int index, UAVObjectField *field, TreeItem *parent)
{
QList<QVariant> data;
if (field->getNumElements() == 1) {
2013-04-05 23:46:56 +03:00
data.append(field->getName());
} else {
data.append(QString("[%1]").arg((field->getElementNames())[index]));
}
2013-04-05 23:46:56 +03:00
FieldTreeItem *item = NULL;
2013-04-05 23:46:56 +03:00
UAVObjectField::FieldType type = field->getType();
// hack: list available actions in an enum
if (field->getName().compare("Action") == 0 && type == UAVObjectField::UINT8) {
data.append(field->getValue(index).toInt());
data.append(field->getUnits());
2013-04-05 23:46:56 +03:00
item = new ActionFieldTreeItem(field, index, data, m_actions);
} else {
switch (type) {
case UAVObjectField::ENUM:
{
QStringList options = field->getOptions();
QVariant value = field->getValue();
data.append(options.indexOf(value.toString()));
data.append(field->getUnits());
item = new EnumFieldTreeItem(field, index, data);
break;
}
case UAVObjectField::INT8:
case UAVObjectField::INT16:
case UAVObjectField::INT32:
case UAVObjectField::UINT8:
case UAVObjectField::UINT16:
case UAVObjectField::UINT32:
data.append(field->getValue(index));
data.append(field->getUnits());
item = new IntFieldTreeItem(field, index, data);
break;
case UAVObjectField::FLOAT32:
data.append(field->getValue(index));
data.append(field->getUnits());
item = new FloatFieldTreeItem(field, index, data);
break;
default:
Q_ASSERT(false);
}
2013-04-05 23:46:56 +03:00
}
connect(item, SIGNAL(updateHighlight(TreeItem *)), this, SLOT(updateHighlight(TreeItem *)));
2013-04-05 23:46:56 +03:00
parent->appendChild(item);
}
QModelIndex PathActionEditorTreeModel::index(int row, int column, const QModelIndex &parent)
const
2013-04-05 23:46:56 +03:00
{
if (!hasIndex(row, column, parent)) {
2013-04-05 23:46:56 +03:00
return QModelIndex();
}
2013-04-05 23:46:56 +03:00
TreeItem *parentItem;
if (!parent.isValid()) {
2013-04-05 23:46:56 +03:00
parentItem = m_rootItem;
} else {
parentItem = static_cast<TreeItem *>(parent.internalPointer());
}
2013-04-05 23:46:56 +03:00
TreeItem *childItem = parentItem->child(row);
if (childItem) {
2013-04-05 23:46:56 +03:00
return createIndex(row, column, childItem);
} else {
2013-04-05 23:46:56 +03:00
return QModelIndex();
}
2013-04-05 23:46:56 +03:00
}
QModelIndex PathActionEditorTreeModel::index(TreeItem *item)
{
if (item->parent() == 0) {
2013-04-05 23:46:56 +03:00
return QModelIndex();
}
2013-04-05 23:46:56 +03:00
QModelIndex root = index(item->parent());
for (int i = 0; i < rowCount(root); ++i) {
QModelIndex childIndex = index(i, 0, root);
TreeItem *child = static_cast<TreeItem *>(childIndex.internalPointer());
if (child == item) {
2013-04-05 23:46:56 +03:00
return childIndex;
}
2013-04-05 23:46:56 +03:00
}
Q_ASSERT(false);
return QModelIndex();
}
QModelIndex PathActionEditorTreeModel::parent(const QModelIndex &index) const
{
if (!index.isValid()) {
2013-04-05 23:46:56 +03:00
return QModelIndex();
}
2013-04-05 23:46:56 +03:00
TreeItem *childItem = static_cast<TreeItem *>(index.internalPointer());
2013-04-05 23:46:56 +03:00
TreeItem *parentItem = childItem->parent();
if (parentItem == m_rootItem) {
2013-04-05 23:46:56 +03:00
return QModelIndex();
}
2013-04-05 23:46:56 +03:00
return createIndex(parentItem->row(), 0, parentItem);
}
int PathActionEditorTreeModel::rowCount(const QModelIndex &parent) const
{
TreeItem *parentItem;
if (parent.column() > 0) {
2013-04-05 23:46:56 +03:00
return 0;
}
2013-04-05 23:46:56 +03:00
if (!parent.isValid()) {
2013-04-05 23:46:56 +03:00
parentItem = m_rootItem;
} else {
parentItem = static_cast<TreeItem *>(parent.internalPointer());
}
2013-04-05 23:46:56 +03:00
return parentItem->childCount();
}
int PathActionEditorTreeModel::columnCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return static_cast<TreeItem *>(parent.internalPointer())->columnCount();
} else {
2013-04-05 23:46:56 +03:00
return m_rootItem->columnCount();
}
2013-04-05 23:46:56 +03:00
}
QVariant PathActionEditorTreeModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid()) {
2013-04-05 23:46:56 +03:00
return QVariant();
}
2013-04-05 23:46:56 +03:00
if (index.column() == TreeItem::dataColumn && role == Qt::EditRole) {
TreeItem *item = static_cast<TreeItem *>(index.internalPointer());
2013-04-05 23:46:56 +03:00
return item->data(index.column());
}
// if (role == Qt::DecorationRole)
// return QIcon(":/core/images/openpilot_logo_128.png");
2013-04-05 23:46:56 +03:00
if (role == Qt::ToolTipRole) {
TreeItem *item = static_cast<TreeItem *>(index.internalPointer());
2013-04-05 23:46:56 +03:00
return item->description();
}
TreeItem *item = static_cast<TreeItem *>(index.internalPointer());
2013-04-05 23:46:56 +03:00
if (index.column() == 0 && role == Qt::BackgroundRole) {
ObjectTreeItem *objItem = dynamic_cast<ObjectTreeItem *>(item);
if (objItem && objItem->highlighted()) {
2013-04-05 23:46:56 +03:00
return QVariant(m_recentlyUpdatedColor);
}
2013-04-05 23:46:56 +03:00
}
if (index.column() == TreeItem::dataColumn && role == Qt::BackgroundRole) {
FieldTreeItem *fieldItem = dynamic_cast<FieldTreeItem *>(item);
if (fieldItem && fieldItem->highlighted()) {
2013-04-05 23:46:56 +03:00
return QVariant(m_recentlyUpdatedColor);
}
if (fieldItem && fieldItem->changed()) {
2013-04-05 23:46:56 +03:00
return QVariant(m_manuallyChangedColor);
}
2013-04-05 23:46:56 +03:00
}
if (role != Qt::DisplayRole) {
2013-04-05 23:46:56 +03:00
return QVariant();
}
2013-04-05 23:46:56 +03:00
if (index.column() == TreeItem::dataColumn) {
EnumFieldTreeItem *fieldItem = dynamic_cast<EnumFieldTreeItem *>(item);
2013-04-05 23:46:56 +03:00
if (fieldItem) {
int enumIndex = fieldItem->data(index.column()).toInt();
return fieldItem->enumOptions(enumIndex);
} else {
ActionFieldTreeItem *afieldItem = dynamic_cast<ActionFieldTreeItem *>(item);
2013-04-05 23:46:56 +03:00
if (afieldItem) {
int enumIndex = afieldItem->data(index.column()).toInt();
return afieldItem->enumOptions(enumIndex);
}
}
2013-04-05 23:46:56 +03:00
}
return item->data(index.column());
}
bool PathActionEditorTreeModel::setData(const QModelIndex &index, const QVariant & value, int role)
{
Q_UNUSED(role)
TreeItem * item = static_cast<TreeItem *>(index.internalPointer());
2013-04-05 23:46:56 +03:00
item->setData(value, index.column());
return true;
}
Qt::ItemFlags PathActionEditorTreeModel::flags(const QModelIndex &index) const
{
if (!index.isValid()) {
2013-04-05 23:46:56 +03:00
return 0;
}
2013-04-05 23:46:56 +03:00
if (index.column() == TreeItem::dataColumn) {
TreeItem *item = static_cast<TreeItem *>(index.internalPointer());
if (item->isEditable()) {
2013-04-05 23:46:56 +03:00
return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable;
}
2013-04-05 23:46:56 +03:00
}
return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
}
QVariant PathActionEditorTreeModel::headerData(int section, Qt::Orientation orientation,
int role) const
2013-04-05 23:46:56 +03:00
{
if (orientation == Qt::Horizontal && role == Qt::DisplayRole) {
2013-04-05 23:46:56 +03:00
return m_rootItem->data(section);
}
2013-04-05 23:46:56 +03:00
return QVariant();
}
void PathActionEditorTreeModel::updateHighlight(TreeItem *item)
{
QModelIndex itemIndex = index(item);
2013-04-05 23:46:56 +03:00
Q_ASSERT(itemIndex != QModelIndex());
emit dataChanged(itemIndex, itemIndex.sibling(itemIndex.row(), TreeItem::dataColumn));
// update uavobject if any - go up the tree until there is
ObjectTreeItem *objItem = 0;
TreeItem *searchItem = item;
2013-04-05 23:46:56 +03:00
while (searchItem) {
objItem = dynamic_cast<ObjectTreeItem *>(searchItem);
if (objItem) {
2013-04-05 23:46:56 +03:00
break;
}
2013-04-05 23:46:56 +03:00
searchItem = searchItem->parent();
}
if (objItem) {
item->apply();
objItem->apply();
2013-04-05 23:46:56 +03:00
UAVObject *obj = objItem->object();
Q_ASSERT(obj);
obj->updated();
}
}
void PathActionEditorTreeModel::highlightUpdatedObject(UAVObject *obj)
{
Q_ASSERT(obj);
if (obj->getName().compare("Waypoint") == 0) {
2013-04-05 23:46:56 +03:00
m_waypointsTree->update();
// emit dataChanged(index(m_waypointsTree), index(m_waypointsTree));
} else if (obj->getName().compare("PathAction") == 0) {
2013-04-05 23:46:56 +03:00
m_pathactionsTree->update();
// emit dataChanged(index(m_pathactionsTree), index(m_pathactionsTree));
2013-04-05 23:46:56 +03:00
}
}
void PathActionEditorTreeModel::newInstance(UAVObject *obj)
{
if (obj->getName().compare("Waypoint") == 0) {
addInstance(obj, m_waypointsTree);
2013-04-05 23:46:56 +03:00
m_waypointsTree->update();
} else if (obj->getName().compare("PathAction") == 0) {
addInstance(obj, m_pathactionsTree);
2013-04-05 23:46:56 +03:00
m_pathactionsTree->update();
}
updateActions();
emit layoutChanged();
}
void PathActionEditorTreeModel::objUpdated(UAVObject *obj)
{
Q_UNUSED(obj);
quint16 index = m_objManager->getObject("WaypointActive")->getField("Index")->getValue().toInt();
quint16 action = 0;
foreach(TreeItem * child, m_waypointsTree->treeChildren()) {
ObjectTreeItem *objItem = dynamic_cast<ObjectTreeItem *>(child);
2013-04-05 23:46:56 +03:00
if (index == objItem->object()->getInstID()) {
child->setActive(true);
action = objItem->object()->getField("Action")->getValue().toInt();
} else {
child->setActive(false);
}
}
foreach(TreeItem * child, m_pathactionsTree->treeChildren()) {
ObjectTreeItem *objItem = dynamic_cast<ObjectTreeItem *>(child);
2013-04-05 23:46:56 +03:00
if (action == objItem->object()->getInstID()) {
child->setActive(true);
} else {
child->setActive(false);
}
}
updateActions();
emit layoutChanged();
}