QGCParamWidget.cc 18.9 KB
Newer Older
1 2 3 4
/*=====================================================================

QGroundControl Open Source Ground Control Station

pixhawk's avatar
pixhawk committed
5
(c) 2009, 2010 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27

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>
 */
Lorenz Meier's avatar
Lorenz Meier committed
28 29
#include <cmath>
#include <float.h>
tstellanova's avatar
tstellanova committed
30 31
#include <QApplication>
#include <QDebug>
32
#include <QFile>
tstellanova's avatar
tstellanova committed
33 34
#include <QGridLayout>

35
#include <QList>
tstellanova's avatar
tstellanova committed
36 37 38
#include <QPushButton>
#include <QSettings>
#include <QTime>
pixhawk's avatar
pixhawk committed
39

Lorenz Meier's avatar
Lorenz Meier committed
40
#include "MainWindow.h"
41
#include "QGC.h"
tstellanova's avatar
tstellanova committed
42 43 44
#include "QGCParamWidget.h"
#include "UASInterface.h"
#include "UASParameterCommsMgr.h"
pixhawk's avatar
pixhawk committed
45

46 47 48 49
/**
 * @param uas MAV to set the parameters on
 * @param parent Parent widget
 */
50 51
QGCParamWidget::QGCParamWidget(QWidget *parent) :
    QGCBaseParamWidget(parent),
52 53
    componentItems(new QMap<int, QTreeWidgetItem*>()),
    statusLabel(new QLabel(this)),
54 55
    tree(new QTreeWidget(this)),
    _fullParamListLoaded(false)
pixhawk's avatar
pixhawk committed
56
{
lm's avatar
lm committed
57

58

59 60 61
}


62

63 64 65 66
void QGCParamWidget::disconnectViewSignalsAndSlots()
{
    disconnect(tree, SIGNAL(itemChanged(QTreeWidgetItem*,int)),
            this, SLOT(parameterItemChanged(QTreeWidgetItem*,int)));
67
}
68 69


70 71
void QGCParamWidget::connectViewSignalsAndSlots()
{
72 73 74 75 76
    // Listen for edits to the tree UI
    connect(tree, SIGNAL(itemChanged(QTreeWidgetItem*,int)),
            this, SLOT(parameterItemChanged(QTreeWidgetItem*,int)));
}

77

78
void QGCParamWidget::addActionButtonsToLayout(QGridLayout* layout)
79
{
Jessica's avatar
Jessica committed
80
    QPushButton* refreshButton = new QPushButton(tr("Get"));
81 82
    refreshButton->setToolTip(tr("Fetch parameters currently in volatile memory of aircraft."));
    refreshButton->setWhatsThis(tr("Fetch parameters currently in volatile memory of aircraft."));
83 84
    connect(refreshButton, SIGNAL(clicked()),
            this, SLOT(requestOnboardParamsUpdate()));
85
    layout->addWidget(refreshButton, 2, 0);
86

Jessica's avatar
Jessica committed
87
    QPushButton* setButton = new QPushButton(tr("Set"));
88 89
    setButton->setToolTip(tr("Send pending parameters to volatile onboard memory"));
    setButton->setWhatsThis(tr("Send pending parameters to volatile onboard memory"));
tstellanova's avatar
tstellanova committed
90
    connect(setButton, SIGNAL(clicked()),
91
            paramMgr, SLOT(sendPendingParameters()));
92
    layout->addWidget(setButton, 2, 1);
93

94
    QPushButton* writeButton = new QPushButton(tr("Write (ROM)"));
95 96
    writeButton->setToolTip(tr("Copy parameters in volatile memory of the aircraft to persistent memory. Transmit your parameters first to write these."));
    writeButton->setWhatsThis(tr("Copy parameters in volatile memory of the aircraft to persistent memory. Transmit your parameters first to write these."));
97 98
    connect(writeButton, SIGNAL(clicked()),
            paramMgr, SLOT(copyVolatileParamsToPersistent()));
99
    layout->addWidget(writeButton, 2, 2);
100

101
    QPushButton* loadFileButton = new QPushButton(tr("Load File"));
102 103
    loadFileButton->setToolTip(tr("Load parameters from a file into qgroundcontrol. To write these to the aircraft, use transmit after loading them."));
    loadFileButton->setWhatsThis(tr("Load parameters from a file into qgroundcontrol. To write these to the aircraft, use transmit after loading them."));
104 105
    connect(loadFileButton, SIGNAL(clicked()),
            this, SLOT(loadParametersFromFile()));
106
    layout->addWidget(loadFileButton, 3, 0);
107 108

    QPushButton* saveFileButton = new QPushButton(tr("Save File"));
109 110
    saveFileButton->setToolTip(tr("Save parameters in this view to a file on this computer."));
    saveFileButton->setWhatsThis(tr("Save parameters in this view to a file on this computer."));
111 112
    connect(saveFileButton, SIGNAL(clicked()),
            this, SLOT(saveParametersToFile()));
113
    layout->addWidget(saveFileButton, 3, 1);
lm's avatar
lm committed
114 115

    QPushButton* readButton = new QPushButton(tr("Read (ROM)"));
116 117
    readButton->setToolTip(tr("Copy parameters from persistent onboard memory to volatile onboard memory of aircraft. DOES NOT update the parameters in this view: click refresh after copying them to get them."));
    readButton->setWhatsThis(tr("Copy parameters from persistent onboard memory to volatile onboard memory of aircraft. DOES NOT update the parameters in this view: click refresh after copying them to get them."));
118 119
    connect(readButton, SIGNAL(clicked()),
            paramMgr, SLOT(copyPersistentParamsToVolatile()));
120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
    layout->addWidget(readButton, 3, 2);

}

void QGCParamWidget::layoutWidget()
{

    statusLabel->setAutoFillBackground(true);

    QGridLayout* layout = new QGridLayout(this);
    layout->setHorizontalSpacing(6);
    layout->setVerticalSpacing(6);
    layout->setMargin(0);
    layout->setSizeConstraint(QLayout::SetMinimumSize);

    // Parameter tree
    layout->addWidget(tree, 0, 0, 1, 3);

    // Status line
    statusLabel->setText(tr("Click refresh to download parameters"));
    layout->addWidget(statusLabel, 1, 0, 1, 3);

    // BUTTONS
    addActionButtonsToLayout(layout);
144

145
    // Set correct vertical scaling
146 147 148 149
    layout->setRowStretch(0, 100);
    layout->setRowStretch(1, 10);
    layout->setRowStretch(2, 10);
    layout->setRowStretch(3, 10);
150

151
    // Set layout
152
    this->setLayout(layout);
pixhawk's avatar
pixhawk committed
153 154 155 156 157 158 159

    // Set header
    QStringList headerItems;
    headerItems.append("Parameter");
    headerItems.append("Value");
    tree->setHeaderLabels(headerItems);
    tree->setColumnCount(2);
tstellanova's avatar
tstellanova committed
160
    tree->setColumnWidth(0,200);
161
    tree->setColumnWidth(1,120);
162
    tree->setExpandsOnDoubleClick(true);
163

164
    tree->setVisible(true);
pixhawk's avatar
pixhawk committed
165 166
}

tstellanova's avatar
tstellanova committed
167

tstellanova's avatar
tstellanova committed
168
void QGCParamWidget::addComponentItem(int compId, QString compName)
pixhawk's avatar
pixhawk committed
169
{
tstellanova's avatar
tstellanova committed
170

171 172
    QString compLine = QString("%1 (#%2)").arg(compName).arg(compId);

173 174
    //QString ptrStr = QString().sprintf("%8p", this);
    //qDebug() <<  "QGCParamWidget" << ptrStr << "addComponentItem:" << compLine;
175

tstellanova's avatar
tstellanova committed
176
    if (componentItems->contains(compId)) {
177 178
        // Update existing component item
        componentItems->value(compId)->setData(0, Qt::DisplayRole, compLine);
179
        //components->value(component)->setData(1, Qt::DisplayRole, QString::number(component));
tstellanova's avatar
tstellanova committed
180
        componentItems->value(compId)->setFirstColumnSpanned(true);
181
    } else {
182 183 184 185 186 187
        // Add new component item
        QStringList list(compLine);
        QTreeWidgetItem* compItem = new QTreeWidgetItem(list);
        compItem->setFirstColumnSpanned(true);
        componentItems->insert(compId, compItem);
        // Create parameter grouping for this component and update maps
tstellanova's avatar
tstellanova committed
188
        paramGroups.insert(compId, new QMap<QString, QTreeWidgetItem*>());
189
        tree->addTopLevelItem(compItem);
pixhawk's avatar
pixhawk committed
190 191
        tree->update();
    }
192

pixhawk's avatar
pixhawk committed
193 194
}

195 196
void QGCParamWidget::handlePendingParamUpdate(int compId, const QString& paramName, QVariant value, bool isPending)
{
197
    //qDebug() << "handlePendingParamUpdate:" << paramName << "with updatingParamNameLock:" << updatingParamNameLock;
198

199
    if (updatingParamNameLock == paramName) {
200
        //qDebug() << "ignoring bounce from " << paramName;
201 202 203 204
        return;
    }
    else {
        updatingParamNameLock = paramName;
205 206
    }

207
    QTreeWidgetItem* paramItem = updateParameterDisplay(compId,paramName,value);
Don Gagne's avatar
Don Gagne committed
208 209 210 211 212 213 214 215 216 217 218
    if (paramItem) {
        if (isPending) {
            paramItem->setBackground(0, QBrush(QColor(QGC::colorOrange)));
            paramItem->setBackground(1, QBrush(QColor(QGC::colorOrange)));
            //ensure that the adjusted item is visible
            tree->expandItem(paramItem->parent());
        }
        else {
            paramItem->setBackground(0, Qt::NoBrush);
            paramItem->setBackground(1, Qt::NoBrush);
        }
219 220
    }

221 222
    updatingParamNameLock.clear();

223
}
tstellanova's avatar
tstellanova committed
224

225
void QGCParamWidget::handleOnboardParamUpdate(int compId, const QString& paramName, QVariant value)
lm's avatar
lm committed
226
{
227
    //qDebug() << "handlePendingParamUpdate:" << paramName << "with updatingParamNameLock:" << updatingParamNameLock;
228
    if (paramName == updatingParamNameLock) {
229
        //qDebug() << "handlePendingParamUpdate ignoring bounce from " << paramName;
230 231
        return;
    }
232
    updatingParamNameLock = paramName;
233
    updateParameterDisplay(compId, paramName, value);
234
    updatingParamNameLock.clear();
tstellanova's avatar
tstellanova committed
235
}
236

237

238
void QGCParamWidget::handleOnboardParameterListUpToDate()
tstellanova's avatar
tstellanova committed
239
{
240 241 242 243 244 245 246
    // Don't load full param list more than once
    if (_fullParamListLoaded) {
        return;
    }
    
    _fullParamListLoaded = true;
    
247 248
    //turn off updates while we refresh the entire list
    tree->setUpdatesEnabled(false);
249 250 251

    //rewrite the component item tree after receiving the full list
    QMap<int, QMap<QString, QVariant>*>::iterator i;
252
    QMap<int, QMap<QString, QVariant>*>* onboardParams = paramMgr->dataModel()->getAllOnboardParams();
253 254 255 256 257 258

    for (i = onboardParams->begin(); i != onboardParams->end(); ++i) {
        int compId = i.key();
        QMap<QString, QVariant>* paramPairs = onboardParams->value(compId);
        QMap<QString, QVariant>::iterator j;
        for (j = paramPairs->begin(); j != paramPairs->end(); j++) {
259
            updatingParamNameLock = j.key();
260
            updateParameterDisplay(compId, j.key(),j.value());
261
            updatingParamNameLock.clear();
262 263 264
        }
    }

tstellanova's avatar
tstellanova committed
265 266
    // Expand visual tree
    tree->expandItem(tree->topLevelItem(0));
267 268 269
    tree->setUpdatesEnabled(true);
    tree->update();

lm's avatar
lm committed
270 271
}

272
QTreeWidgetItem* QGCParamWidget::findChildWidgetItemForParam(QTreeWidgetItem* parentItem, const QString& paramName)
tstellanova's avatar
tstellanova committed
273
{
274 275 276 277 278 279 280 281 282
    QTreeWidgetItem* childItem = NULL;

    for (int i = 0; i < parentItem->childCount(); i++) {
        QTreeWidgetItem* child = parentItem->child(i);
        QString key = child->data(0, Qt::DisplayRole).toString();
        if (key == paramName)  {
            childItem = child;
            break;
        }
pixhawk's avatar
pixhawk committed
283
    }
284

285 286 287 288 289
    return childItem;
}

QTreeWidgetItem* QGCParamWidget::getParentWidgetItemForParam(int compId, const QString& paramName)
{
290 291
    QTreeWidgetItem* parentItem = componentItems->value(compId);

pixhawk's avatar
pixhawk committed
292 293
    QString splitToken = "_";
    // Check if auto-grouping can work
294 295
    if (paramName.contains(splitToken)) {
        QString parentStr = paramName.section(splitToken, 0, 0, QString::SectionSkipEmpty);
tstellanova's avatar
tstellanova committed
296 297
        QMap<QString, QTreeWidgetItem*>* compParamGroups = paramGroups.value(compId);
        if (!compParamGroups->contains(parentStr)) {
pixhawk's avatar
pixhawk committed
298 299
            // Insert group item
            QStringList glist;
tstellanova's avatar
tstellanova committed
300
            glist.append(parentStr);
301
            QTreeWidgetItem* groupItem = new QTreeWidgetItem(glist);
tstellanova's avatar
tstellanova committed
302 303 304 305 306 307 308

            compParamGroups->insert(parentStr, groupItem);

            // insert new group alphabetized
            QList<QString> groupKeys = compParamGroups->uniqueKeys();
            int insertIdx = groupKeys.indexOf(parentStr);
            componentItems->value(compId)->insertChild(insertIdx,groupItem);
309
        }
310

311
        //parent item for this parameter item will be a group widget item
312 313 314
        parentItem = compParamGroups->value(parentStr);
    }
    else  {
315
        //parent item for this parameter will be the top level (component) widget item
316 317 318
        parentItem = componentItems->value(compId);
    }

319 320
    return parentItem;
}
321

322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341
void QGCParamWidget::insertParamAlphabetical(int indexLowerBound, int indexUpperBound, QTreeWidgetItem* parentItem, QTreeWidgetItem* paramItem)
{
    if (indexLowerBound >= indexUpperBound)
    {
        if (paramItem->text(0).compare(parentItem->child(indexLowerBound)->text(0)) < 0) {
            parentItem->insertChild(indexLowerBound, paramItem);
        }
        else
        {
            if (indexLowerBound < parentItem->childCount() - 1) {
                parentItem->insertChild(indexLowerBound + 1, paramItem);
            }
            else
            {
                parentItem->addChild(paramItem);
            }
        }
    }
    else
    {
Don Gagne's avatar
Don Gagne committed
342
        int midpoint = indexLowerBound + ((indexUpperBound - indexLowerBound) / 2);
343 344 345 346 347 348 349 350 351 352 353 354

        if (paramItem->text(0).compare(parentItem->child(midpoint)->text(0)) < 0)
        {
            insertParamAlphabetical(indexLowerBound, midpoint - 1, parentItem, paramItem);
        } else
        {
            insertParamAlphabetical(midpoint + 1, indexUpperBound, parentItem, paramItem);
        }

    }
}

355 356
QTreeWidgetItem* QGCParamWidget::updateParameterDisplay(int compId, QString parameterName, QVariant value)
{
357
    //qDebug() << "QGCParamWidget::updateParameterDisplay" << parameterName;
Don Gagne's avatar
Don Gagne committed
358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375
    
    // Filter the parameters according to the filter list
    if (_filterList.count() != 0) {
        bool filterFound = false;
        foreach(QString paramFilter, _filterList) {
            if (paramFilter.endsWith("*") && parameterName.startsWith(paramFilter.left(paramFilter.size() - 1))) {
                filterFound = true;
                break;
            }
            if (paramFilter == parameterName) {
                filterFound = true;
                break;            
            }
        }
        if (!filterFound) {
            return NULL;
        }
    }
376 377

    // Reference to item in tree
378
    QTreeWidgetItem* paramItem = NULL;
379 380 381 382 383 384 385 386 387 388

    // Add component item if necessary
    if (!componentItems->contains(compId)) {
        QString componentName = tr("Component #%1").arg(compId);
        addComponentItem(compId, componentName);
    }

    //default parent item for this parameter widget item will be the top level component item
    QTreeWidgetItem* parentItem = getParentWidgetItemForParam(compId,parameterName);
    if (parentItem) {
389 390
        paramItem = findChildWidgetItemForParam(parentItem,parameterName);
        if (!paramItem) {
391 392 393
            // Insert parameter into map
            QStringList plist;
            plist.append(parameterName);
394
            // CREATE PARAMETER ITEM
395
            paramItem = new QTreeWidgetItem(plist);
396
            // CONFIGURE PARAMETER ITEM
tstellanova's avatar
tstellanova committed
397
            if (value.type() == QVariant::Char) {
398
                paramItem->setData(1, Qt::DisplayRole, value.toUInt());
399
            }
tstellanova's avatar
tstellanova committed
400
            else {
401
                paramItem->setData(1, Qt::DisplayRole, value);
402
            }
403
            paramItem->setFlags(paramItem->flags() | Qt::ItemIsEditable);
404

405
            //Insert alphabetically
406 407 408
            if (parentItem->childCount() > 0) {
                insertParamAlphabetical(0, parentItem->childCount() - 1, parentItem, paramItem);
            } else
409 410 411
            {
                parentItem->addChild(paramItem);
            }
412 413

            //only add the tooltip when the parameter item is first added
414
            QString paramDesc = paramMgr->dataModel()->getParamDescription(parameterName);
415 416
            if (!paramDesc.isEmpty()) {
                QString tooltipFormat;
417
                if (paramMgr->dataModel()->isParamDefaultKnown(parameterName)) {
418
                    tooltipFormat = tr("Default: %1, %2");
419
                    double paramDefValue = paramMgr->dataModel()->getParamDefault(parameterName);
420 421 422 423 424
                    tooltipFormat = tooltipFormat.arg(paramDefValue).arg(paramDesc);
                }
                else {
                    tooltipFormat = paramDesc;
                }
425 426
                paramItem->setToolTip(0, tooltipFormat);
                paramItem->setToolTip(1, tooltipFormat);
427 428 429
            }
        }

430
        //update the parameterItem's data
431
        if (value.type() == QVariant::Char) {
432
            paramItem->setData(1, Qt::DisplayRole, value.toUInt());
433 434
        }
        else {
435
            paramItem->setData(1, Qt::DisplayRole, value);
pixhawk's avatar
pixhawk committed
436
        }
437

438
    }
439

440
    if (paramItem) {
441
        // Reset background color
442 443
        paramItem->setBackground(0, Qt::NoBrush);
        paramItem->setBackground(1, Qt::NoBrush);
444

445 446 447 448 449 450 451
        paramItem->setTextColor(0, QGC::colorDarkWhite);
        paramItem->setTextColor(1, QGC::colorDarkWhite);

        if (paramItem == tree->currentItem()) {
            //need to unset current item to clear highlighting (green by default)
            tree->setCurrentItem(NULL); //clear the selected line
        }
452

453
    }
454

455
    return paramItem;
pixhawk's avatar
pixhawk committed
456 457
}

lm's avatar
lm committed
458

459

460
void QGCParamWidget::parameterItemChanged(QTreeWidgetItem* paramItem, int column)
lm's avatar
lm committed
461
{
462
    if (paramItem && column > 0) {
463

464
        QString key = paramItem->data(0, Qt::DisplayRole).toString();
465
        //qDebug() << "parameterItemChanged:" << key << "with updatingParamNameLock:" << updatingParamNameLock;
466 467

        if (key == updatingParamNameLock) {
468
            //qDebug() << "parameterItemChanged ignoring bounce from " << key;
469 470
            return;
        }
471 472 473
        else {
            updatingParamNameLock = key;
        }
474 475

        QTreeWidgetItem* parent = paramItem->parent();
476
        while (parent->parent() != NULL) {
477 478 479
            parent = parent->parent();
        }
        // Parent is now top-level component
480
        int componentId = componentItems->key(parent);
481
        QVariant value = paramItem->data(1, Qt::DisplayRole);
482

483

484
        bool pending = paramMgr->dataModel()->updatePendingParamWithValue(componentId,key,value);
485

486 487
        // If the value will result in an update
        if (pending) {
488
            // Set parameter on changed list to be transmitted to MAV
489
            statusLabel->setText(tr("Pending: %1:%2: %3").arg(componentId).arg(key).arg(value.toFloat(), 5, 'f', 1, QChar(' ')));
490

491 492
            paramItem->setBackground(0, QBrush(QColor(QGC::colorOrange)));
            paramItem->setBackground(1, QBrush(QColor(QGC::colorOrange)));
493
        }
494
        else {
495
            QMap<QString , QVariant>* pendingParams = paramMgr->dataModel()->getPendingParamsForComponent(componentId);
496 497
            int pendingCount = pendingParams->count();
            statusLabel->setText(tr("Pending items: %1").arg(pendingCount));
498 499
            paramItem->setBackground(0, Qt::NoBrush);
            paramItem->setBackground(1, Qt::NoBrush);
500
        }
501

502 503 504 505 506 507

        if (paramItem == tree->currentItem()) {
            //need to unset current item to clear highlighting (green by default)
            tree->setCurrentItem(NULL); //clear the selected line
        }

508
        updatingParamNameLock.clear();
509 510 511
    }
}

512

tstellanova's avatar
tstellanova committed
513
void QGCParamWidget::setParameterStatusMsg(const QString& msg)
514
{
tstellanova's avatar
tstellanova committed
515 516 517
    statusLabel->setText(msg);
}

518

519
void QGCParamWidget::clearOnboardParamDisplay()
520
{
521 522
    tree->clear();
    componentItems->clear();
523 524
}

525
void QGCParamWidget::clearPendingParamDisplay()
pixhawk's avatar
pixhawk committed
526 527
{
    tree->clear();
528
    componentItems->clear();
pixhawk's avatar
pixhawk committed
529
}
tstellanova's avatar
tstellanova committed
530

531

tstellanova's avatar
tstellanova committed
532 533 534 535 536 537 538 539 540 541 542 543 544 545 546
void QGCParamWidget::handleParamStatusMsgUpdate(QString msg, int level)
{
    QColor bgColor = QGC::colorGreen;
    if ((int)UASParameterCommsMgr::ParamCommsStatusLevel_Warning == level) {
        bgColor = QGC::colorOrange;
    }
    else if ((int)UASParameterCommsMgr::ParamCommsStatusLevel_Error == level) {
        bgColor =  QGC::colorRed;
    }

    QPalette pal = statusLabel->palette();
    pal.setColor(backgroundRole(), bgColor);
    statusLabel->setPalette(pal);
    statusLabel->setText(msg);
}