QGCParamWidget.cc 10.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
/*=====================================================================

QGroundControl Open Source Ground Control Station

(c) 2009, 2010 QGROUNDCONTROL/PIXHAWK PROJECT
<http://www.qgroundcontrol.org>
<http://pixhawk.ethz.ch>

This file is part of the QGROUNDCONTROL project

    QGROUNDCONTROL 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
    (at your option) any later version.

    QGROUNDCONTROL 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 QGROUNDCONTROL. If not, see <http://www.gnu.org/licenses/>.

======================================================================*/

/**
 * @file
 *   @brief Implementation of class QGCParamWidget
 *   @author Lorenz Meier <mail@qgroundcontrol.org>
 */

32 33
#include <QGridLayout>
#include <QPushButton>
pixhawk's avatar
pixhawk committed
34 35 36 37

#include "QGCParamWidget.h"
#include "UASInterface.h"
#include <QDebug>
38
#include "QGC.h"
pixhawk's avatar
pixhawk committed
39

40 41 42 43
/**
 * @param uas MAV to set the parameters on
 * @param parent Parent widget
 */
pixhawk's avatar
pixhawk committed
44
QGCParamWidget::QGCParamWidget(UASInterface* uas, QWidget *parent) :
45 46
        QWidget(parent),
        mav(uas),
lm's avatar
lm committed
47
        components(new QMap<int, QTreeWidgetItem*>()),
pixhawk's avatar
pixhawk committed
48 49
        paramGroups(),
        changedValues()
pixhawk's avatar
pixhawk committed
50 51 52
{
    // Create tree widget
    tree = new QTreeWidget(this);
53
    tree->setColumnWidth(0, 150);
pixhawk's avatar
pixhawk committed
54 55

    // Set tree widget as widget onto this component
56
    QGridLayout* horizontalLayout;
pixhawk's avatar
pixhawk committed
57
    //form->setAutoFillBackground(false);
58 59
    horizontalLayout = new QGridLayout(this);
    horizontalLayout->setSpacing(6);
pixhawk's avatar
pixhawk committed
60 61 62
    horizontalLayout->setMargin(0);
    horizontalLayout->setSizeConstraint(QLayout::SetMinimumSize);

63 64 65 66 67 68 69 70 71 72 73 74 75 76
    horizontalLayout->addWidget(tree, 0, 0, 1, 3);
    QPushButton* readButton = new QPushButton(tr("Read"));
    connect(readButton, SIGNAL(clicked()), this, SLOT(requestParameterList()));
    horizontalLayout->addWidget(readButton, 1, 0);

    QPushButton* setButton = new QPushButton(tr("Set (RAM)"));
    connect(setButton, SIGNAL(clicked()), this, SLOT(setParameters()));
    horizontalLayout->addWidget(setButton, 1, 1);

    QPushButton* writeButton = new QPushButton(tr("Write (Disk)"));
    connect(writeButton, SIGNAL(clicked()), this, SLOT(writeParameters()));
    horizontalLayout->addWidget(writeButton, 1, 2);

    // Set layout
pixhawk's avatar
pixhawk committed
77 78 79 80 81 82 83 84
    this->setLayout(horizontalLayout);

    // Set header
    QStringList headerItems;
    headerItems.append("Parameter");
    headerItems.append("Value");
    tree->setHeaderLabels(headerItems);
    tree->setColumnCount(2);
85
    tree->setExpandsOnDoubleClick(true);
86 87 88

    // Connect signals/slots
    connect(this, SIGNAL(parameterChanged(int,QString,float)), mav, SLOT(setParameter(int,QString,float)));
89
    connect(tree, SIGNAL(itemChanged(QTreeWidgetItem*,int)), this, SLOT(parameterItemChanged(QTreeWidgetItem*,int)));
lm's avatar
lm committed
90

91 92
    // New parameters from UAS
    connect(uas, SIGNAL(parameterChanged(int,int,QString,float)), this, SLOT(addParameter(int,int,QString,float)));
pixhawk's avatar
pixhawk committed
93 94
}

95 96 97 98
/**
 * @return The MAV of this widget. Unless the MAV object has been destroyed, this
 *         pointer is never zero.
 */
pixhawk's avatar
pixhawk committed
99 100 101 102 103 104 105 106 107 108 109
UASInterface* QGCParamWidget::getUAS()
{
    return mav;
}

/**
 *
 * @param uas System which has the component
 * @param component id of the component
 * @param componentName human friendly name of the component
 */
110
void QGCParamWidget::addComponent(int uas, int component, QString componentName)
pixhawk's avatar
pixhawk committed
111
{
112
    Q_UNUSED(uas);
pixhawk's avatar
pixhawk committed
113
    if (components->contains(component))
pixhawk's avatar
pixhawk committed
114
    {
pixhawk's avatar
pixhawk committed
115 116 117 118 119 120 121 122 123 124 125 126 127 128
        // Update existing
        components->value(component)->setData(0, Qt::DisplayRole, componentName);
        components->value(component)->setData(1, Qt::DisplayRole, QString::number(component));
    }
    else
    {
        // Add new
        QStringList list;
        list.append(componentName);
        list.append(QString::number(component));
        QTreeWidgetItem* comp = new QTreeWidgetItem(list);
        components->insert(component, comp);
        // Create grouping and update maps
        paramGroups.insert(component, new QMap<QString, QTreeWidgetItem*>());
pixhawk's avatar
pixhawk committed
129 130 131 132 133
        tree->addTopLevelItem(comp);
        tree->update();
    }
}

134 135 136 137 138 139
/**
 * @param uas System which has the component
 * @param component id of the component
 * @param parameterName human friendly name of the parameter
 */
void QGCParamWidget::addParameter(int uas, int component, QString parameterName, float value)
pixhawk's avatar
pixhawk committed
140
{
141
    Q_UNUSED(uas);
142 143
    // Reference to item in tree
    QTreeWidgetItem* parameterItem;
pixhawk's avatar
pixhawk committed
144 145 146 147

    // Get component
    if (!components->contains(component))
    {
148
        addComponent(uas, component, "Component #" + QString::number(component));
pixhawk's avatar
pixhawk committed
149
    }
150

pixhawk's avatar
pixhawk committed
151 152 153
    QString splitToken = "_";
    // Check if auto-grouping can work
    if (parameterName.contains(splitToken))
154
    {
pixhawk's avatar
pixhawk committed
155 156 157
        QString parent = parameterName.section(splitToken, 0, 0, QString::SectionSkipEmpty);
        QMap<QString, QTreeWidgetItem*>* compParamGroups = paramGroups.value(component);
        if (!compParamGroups->contains(parent))
158
        {
pixhawk's avatar
pixhawk committed
159 160 161 162 163 164
            // Insert group item
            QStringList glist;
            glist.append(parent);
            QTreeWidgetItem* item = new QTreeWidgetItem(glist);
            compParamGroups->insert(parent, item);
            components->value(component)->addChild(item);
165
        }
166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191

        // Append child to group
        bool found = false;
        QTreeWidgetItem* parentItem = compParamGroups->value(parent);
        for (int i = 0; i < parentItem->childCount(); i++)
        {
            QTreeWidgetItem* child = parentItem->child(i);
            QString key = child->data(0, Qt::DisplayRole).toString();
            if (key == parameterName)
            {
                //qDebug() << "UPDATED CHILD";
                parameterItem = child;
                parameterItem->setData(1, Qt::DisplayRole, value);
                found = true;
            }
        }

        if (!found)
        {
            // Insert parameter into map
            QStringList plist;
            plist.append(parameterName);
            plist.append(QString::number(value));
            parameterItem = new QTreeWidgetItem(plist);

            compParamGroups->value(parent)->addChild(parameterItem);
lm's avatar
lm committed
192
            parameterItem->setFlags(parameterItem->flags() | Qt::ItemIsEditable);
193
        }
194
    }
pixhawk's avatar
pixhawk committed
195
    else
196
    {
pixhawk's avatar
pixhawk committed
197 198 199 200 201 202 203 204 205
        bool found = false;
        QTreeWidgetItem* parent = components->value(component);
        for (int i = 0; i < parent->childCount(); i++)
        {
            QTreeWidgetItem* child = parent->child(i);
            QString key = child->data(0, Qt::DisplayRole).toString();
            if (key == parameterName)
            {
                //qDebug() << "UPDATED CHILD";
206 207
                parameterItem = child;
                parameterItem->setData(1, Qt::DisplayRole, value);
pixhawk's avatar
pixhawk committed
208 209 210 211 212 213
                found = true;
            }
        }

        if (!found)
        {
214 215 216 217 218 219
            // Insert parameter into map
            QStringList plist;
            plist.append(parameterName);
            plist.append(QString::number(value));
            parameterItem = new QTreeWidgetItem(plist);

lm's avatar
lm committed
220 221
            components->value(component)->addChild(parameterItem);
            parameterItem->setFlags(parameterItem->flags() | Qt::ItemIsEditable);
pixhawk's avatar
pixhawk committed
222
        }
223
        //tree->expandAll();
224
    }
225
    // Reset background color
226 227 228 229
    parameterItem->setBackground(0, QBrush(QColor(0, 0, 0)));
    parameterItem->setBackground(1, Qt::NoBrush);
    //tree->update();
    if (changedValues.contains(component)) changedValues.value(component)->remove(parameterName);
pixhawk's avatar
pixhawk committed
230 231
}

232 233 234 235
/**
 * Send a request to deliver the list of onboard parameters
 * to the MAV.
 */
236 237
void QGCParamWidget::requestParameterList()
{
lm's avatar
lm committed
238
    // Clear view and request param list
239
    clear();
lm's avatar
lm committed
240
    mav->requestParameters();
241 242
}

243
void QGCParamWidget::parameterItemChanged(QTreeWidgetItem* current, int column)
lm's avatar
lm committed
244
{
245
    if (current && column > 0)
lm's avatar
lm committed
246
    {
247 248 249 250 251 252 253 254 255 256 257 258 259
        QTreeWidgetItem* parent = current->parent();
        while (parent->parent() != NULL)
        {
            parent = parent->parent();
        }
        // Parent is now top-level component
        int key = components->key(parent);
        if (!changedValues.contains(key))
        {
            changedValues.insert(key, new QMap<QString, float>());
        }
        QMap<QString, float>* map = changedValues.value(key, NULL);
        if (map)
lm's avatar
lm committed
260
        {
261 262 263 264 265
            bool ok;
            QString str = current->data(0, Qt::DisplayRole).toString();
            float value = current->data(1, Qt::DisplayRole).toDouble(&ok);
            // Send parameter to MAV
            if (ok)
lm's avatar
lm committed
266
            {
267 268 269
                if (ok)
                {
                    qDebug() << "PARAM CHANGED: COMP:" << key << "KEY:" << str << "VALUE:" << value;
270
                    if (map->contains(str)) map->remove(str);
271
                    map->insert(str, value);
272 273
                    //current->setBackground(0, QBrush(QColor(QGC::colorGreen)));
                    //current->setBackground(1, QBrush(QColor(QGC::colorGreen)));
274
                }
lm's avatar
lm committed
275 276
            }
        }
277
    }
lm's avatar
lm committed
278 279
}

280 281 282 283 284 285 286
/**
 * @param component the subsystem which has the parameter
 * @param parameterName name of the parameter, as delivered by the system
 * @param value value of the parameter
 */
void QGCParamWidget::setParameter(int component, QString parameterName, float value)
{
287
    emit parameterChanged(component, parameterName, value);
288
    qDebug() << "SENT PARAM" << parameterName << value;
289 290
}

291 292 293
/**
 * Set all parameter in the parameter tree on the MAV
 */
294 295
void QGCParamWidget::setParameters()
{
296 297 298 299 300 301 302 303 304 305 306
    // Iterate through all components, through all parameters and emit them
    QMap<int, QMap<QString, float>*>::iterator i;
    for (i = changedValues.begin(); i != changedValues.end(); ++i)
    {
        // Iterate through the parameters of the component
        int compid = i.key();
        QMap<QString, float>* comp = i.value();
        {
            QMap<QString, float>::iterator j;
            for (j = comp->begin(); j != comp->end(); ++j)
            {
307
                setParameter(compid, j.key(), j.value());
308 309 310 311
            }
        }
    }

lm's avatar
lm committed
312
    changedValues.clear();
lm's avatar
lm committed
313
    qDebug() << __FILE__ << __LINE__ << "SETTING ALL PARAMETERS";
314 315
}

316 317 318 319
/**
 * Write the current onboard parameters from RAM into
 * permanent storage, e.g. EEPROM or harddisk
 */
320 321 322 323 324
void QGCParamWidget::writeParameters()
{
    mav->writeParameters();
}

325 326 327
/**
 * Clear all data in the parameter widget
 */
pixhawk's avatar
pixhawk committed
328 329 330
void QGCParamWidget::clear()
{
    tree->clear();
lm's avatar
lm committed
331
    components->clear();
pixhawk's avatar
pixhawk committed
332
}