mirror of
https://bitbucket.org/librepilot/librepilot.git
synced 2025-01-25 10:52:11 +01:00
454 lines
17 KiB
C++
454 lines
17 KiB
C++
/**
|
|
******************************************************************************
|
|
*
|
|
* @file configstabilizationwidget.cpp
|
|
* @author E. Lafargue & The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
* @addtogroup GCSPlugins GCS Plugins
|
|
* @{
|
|
* @addtogroup ConfigPlugin Config Plugin
|
|
* @{
|
|
* @brief The Configuration Gadget used to update settings in the firmware
|
|
*****************************************************************************/
|
|
/*
|
|
* 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
|
|
* (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
|
|
* 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.,
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
#include "configstabilizationwidget.h"
|
|
|
|
#include <QDebug>
|
|
#include <QStringList>
|
|
#include <QWidget>
|
|
#include <QTextEdit>
|
|
#include <QVBoxLayout>
|
|
#include <QPushButton>
|
|
#include <QDesktopServices>
|
|
#include <QUrl>
|
|
#include <QList>
|
|
#include <QTabBar>
|
|
#include <QMessageBox>
|
|
|
|
#include <extensionsystem/pluginmanager.h>
|
|
#include <coreplugin/generalsettings.h>
|
|
#include "altitudeholdsettings.h"
|
|
#include "stabilizationsettings.h"
|
|
|
|
#include "qwt/src/qwt.h"
|
|
#include "qwt/src/qwt_plot.h"
|
|
#include "qwt/src/qwt_plot_canvas.h"
|
|
|
|
ConfigStabilizationWidget::ConfigStabilizationWidget(QWidget *parent) : ConfigTaskWidget(parent),
|
|
boardModel(0), m_pidBankCount(0), m_currentPIDBank(0)
|
|
{
|
|
ui = new Ui_StabilizationWidget();
|
|
ui->setupUi(this);
|
|
|
|
setupExpoPlot();
|
|
|
|
StabilizationSettings *stabSettings = qobject_cast<StabilizationSettings *>(getObject("StabilizationSettings"));
|
|
Q_ASSERT(stabSettings);
|
|
|
|
m_pidBankCount = stabSettings->getField("FlightModeMap")->getOptions().count();
|
|
|
|
// Set up fake tab widget stuff for pid banks support
|
|
m_pidTabBars.append(ui->basicPIDBankTabBar);
|
|
m_pidTabBars.append(ui->advancedPIDBankTabBar);
|
|
foreach(QTabBar * tabBar, m_pidTabBars) {
|
|
for (int i = 0; i < m_pidBankCount; i++) {
|
|
tabBar->addTab(tr("Settings Bank %1").arg(i + 1));
|
|
tabBar->setTabData(i, QString("StabilizationSettingsBank%1").arg(i + 1));
|
|
}
|
|
tabBar->setExpanding(false);
|
|
connect(tabBar, SIGNAL(currentChanged(int)), this, SLOT(pidBankChanged(int)));
|
|
}
|
|
|
|
for (int i = 0; i < m_pidBankCount; i++) {
|
|
if (i > 0) {
|
|
m_stabilizationObjectsString.append(",");
|
|
}
|
|
m_stabilizationObjectsString.append(m_pidTabBars.at(0)->tabData(i).toString());
|
|
}
|
|
|
|
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
|
|
Core::Internal::GeneralSettings *settings = pm->getObject<Core::Internal::GeneralSettings>();
|
|
|
|
if (!settings->useExpertMode()) {
|
|
ui->saveStabilizationToRAM_6->setVisible(false);
|
|
}
|
|
|
|
autoLoadWidgets();
|
|
|
|
realtimeUpdates = new QTimer(this);
|
|
connect(realtimeUpdates, SIGNAL(timeout()), this, SLOT(apply()));
|
|
|
|
connect(ui->realTimeUpdates_6, SIGNAL(toggled(bool)), this, SLOT(realtimeUpdatesSlot(bool)));
|
|
addWidget(ui->realTimeUpdates_6);
|
|
connect(ui->realTimeUpdates_8, SIGNAL(toggled(bool)), this, SLOT(realtimeUpdatesSlot(bool)));
|
|
addWidget(ui->realTimeUpdates_8);
|
|
connect(ui->realTimeUpdates_12, SIGNAL(toggled(bool)), this, SLOT(realtimeUpdatesSlot(bool)));
|
|
addWidget(ui->realTimeUpdates_12);
|
|
connect(ui->realTimeUpdates_7, SIGNAL(toggled(bool)), this, SLOT(realtimeUpdatesSlot(bool)));
|
|
addWidget(ui->realTimeUpdates_7);
|
|
|
|
connect(ui->checkBox_7, SIGNAL(toggled(bool)), this, SLOT(linkCheckBoxes(bool)));
|
|
addWidget(ui->checkBox_7);
|
|
connect(ui->checkBox_2, SIGNAL(toggled(bool)), this, SLOT(linkCheckBoxes(bool)));
|
|
addWidget(ui->checkBox_2);
|
|
connect(ui->checkBox_8, SIGNAL(toggled(bool)), this, SLOT(linkCheckBoxes(bool)));
|
|
addWidget(ui->checkBox_8);
|
|
connect(ui->checkBox_3, SIGNAL(toggled(bool)), this, SLOT(linkCheckBoxes(bool)));
|
|
addWidget(ui->checkBox_3);
|
|
|
|
addWidget(ui->pushButton_2);
|
|
addWidget(ui->pushButton_3);
|
|
addWidget(ui->pushButton_4);
|
|
addWidget(ui->pushButton_5);
|
|
addWidget(ui->pushButton_6);
|
|
addWidget(ui->pushButton_7);
|
|
addWidget(ui->pushButton_8);
|
|
addWidget(ui->pushButton_9);
|
|
addWidget(ui->pushButton_10);
|
|
addWidget(ui->pushButton_11);
|
|
addWidget(ui->pushButton_20);
|
|
addWidget(ui->pushButton_22);
|
|
addWidget(ui->pushButton_23);
|
|
|
|
addWidget(ui->basicResponsivenessGroupBox);
|
|
addWidget(ui->basicResponsivenessCheckBox);
|
|
connect(ui->basicResponsivenessCheckBox, SIGNAL(toggled(bool)), this, SLOT(linkCheckBoxes(bool)));
|
|
addWidget(ui->advancedResponsivenessGroupBox);
|
|
addWidget(ui->advancedResponsivenessCheckBox);
|
|
connect(ui->advancedResponsivenessCheckBox, SIGNAL(toggled(bool)), this, SLOT(linkCheckBoxes(bool)));
|
|
|
|
connect(ui->defaultThrottleCurveButton, SIGNAL(clicked()), this, SLOT(resetThrottleCurveToDefault()));
|
|
connect(ui->enableThrustPIDScalingCheckBox, SIGNAL(toggled(bool)), ui->ThrustPIDSource, SLOT(setEnabled(bool)));
|
|
connect(ui->enableThrustPIDScalingCheckBox, SIGNAL(toggled(bool)), ui->ThrustPIDTarget, SLOT(setEnabled(bool)));
|
|
connect(ui->enableThrustPIDScalingCheckBox, SIGNAL(toggled(bool)), ui->ThrustPIDAxis, SLOT(setEnabled(bool)));
|
|
connect(ui->enableThrustPIDScalingCheckBox, SIGNAL(toggled(bool)), ui->thrustPIDScalingCurve, SLOT(setEnabled(bool)));
|
|
ui->thrustPIDScalingCurve->setXAxisLabel(tr("Thrust"));
|
|
ui->thrustPIDScalingCurve->setYAxisLabel(tr("Scaling factor"));
|
|
ui->thrustPIDScalingCurve->setMin(-0.5);
|
|
ui->thrustPIDScalingCurve->setMax(0.5);
|
|
ui->thrustPIDScalingCurve->initLinearCurve(5, -0.25, 0.25);
|
|
connect(ui->thrustPIDScalingCurve, SIGNAL(curveUpdated()), this, SLOT(throttleCurveUpdated()));
|
|
|
|
addWidget(ui->defaultThrottleCurveButton);
|
|
addWidget(ui->enableThrustPIDScalingCheckBox);
|
|
addWidget(ui->thrustPIDScalingCurve);
|
|
addWidget(ui->thrustPIDScalingCurve);
|
|
connect(this, SIGNAL(widgetContentsChanged(QWidget *)), this, SLOT(processLinkedWidgets(QWidget *)));
|
|
|
|
connect(this, SIGNAL(autoPilotConnected()), this, SLOT(onBoardConnected()));
|
|
|
|
connect(ui->expoSpinnerRoll, SIGNAL(valueChanged(int)), this, SLOT(replotExpoRoll(int)));
|
|
connect(ui->expoSpinnerPitch, SIGNAL(valueChanged(int)), this, SLOT(replotExpoPitch(int)));
|
|
connect(ui->expoSpinnerYaw, SIGNAL(valueChanged(int)), this, SLOT(replotExpoYaw(int)));
|
|
|
|
disableMouseWheelEvents();
|
|
updateEnableControls();
|
|
}
|
|
|
|
ConfigStabilizationWidget::~ConfigStabilizationWidget()
|
|
{
|
|
// Do nothing
|
|
}
|
|
|
|
void ConfigStabilizationWidget::refreshWidgetsValues(UAVObject *o)
|
|
{
|
|
ConfigTaskWidget::refreshWidgetsValues(o);
|
|
|
|
updateThrottleCurveFromObject();
|
|
|
|
ui->basicResponsivenessCheckBox->setChecked(ui->rateRollKp_3->value() == ui->ratePitchKp_4->value() &&
|
|
ui->rateRollKi_3->value() == ui->ratePitchKi_4->value());
|
|
}
|
|
|
|
void ConfigStabilizationWidget::updateObjectsFromWidgets()
|
|
{
|
|
updateObjectFromThrottleCurve();
|
|
ConfigTaskWidget::updateObjectsFromWidgets();
|
|
}
|
|
|
|
void ConfigStabilizationWidget::updateThrottleCurveFromObject()
|
|
{
|
|
bool dirty = isDirty();
|
|
UAVObject *stabBank = getObjectManager()->getObject(QString(m_pidTabBars.at(0)->tabData(m_currentPIDBank).toString()));
|
|
|
|
Q_ASSERT(stabBank);
|
|
qDebug() << "updatingCurveFromObject" << stabBank->getName();
|
|
|
|
UAVObjectField *field = stabBank->getField("ThrustPIDScaleCurve");
|
|
Q_ASSERT(field);
|
|
|
|
QList<double> curve;
|
|
for (quint32 i = 0; i < field->getNumElements(); i++) {
|
|
qDebug() << field->getName() << field->getElementNames().at(i) << "=>" << field->getValue(i);
|
|
curve.append(field->getValue(i).toDouble());
|
|
}
|
|
|
|
ui->thrustPIDScalingCurve->setCurve(&curve);
|
|
|
|
field = stabBank->getField("EnableThrustPIDScaling");
|
|
Q_ASSERT(field);
|
|
|
|
bool enabled = field->getValue() == "TRUE";
|
|
ui->enableThrustPIDScalingCheckBox->setChecked(enabled);
|
|
ui->thrustPIDScalingCurve->setEnabled(enabled);
|
|
setDirty(dirty);
|
|
}
|
|
|
|
void ConfigStabilizationWidget::updateObjectFromThrottleCurve()
|
|
{
|
|
UAVObject *stabBank = getObjectManager()->getObject(QString(m_pidTabBars.at(0)->tabData(m_currentPIDBank).toString()));
|
|
|
|
Q_ASSERT(stabBank);
|
|
qDebug() << "updatingObjectFromCurve" << stabBank->getName();
|
|
|
|
UAVObjectField *field = stabBank->getField("ThrustPIDScaleCurve");
|
|
Q_ASSERT(field);
|
|
|
|
QList<double> curve = ui->thrustPIDScalingCurve->getCurve();
|
|
for (quint32 i = 0; i < field->getNumElements(); i++) {
|
|
field->setValue(curve.at(i), i);
|
|
qDebug() << field->getName() << field->getElementNames().at(i) << "<=" << curve.at(i);
|
|
}
|
|
|
|
field = stabBank->getField("EnableThrustPIDScaling");
|
|
Q_ASSERT(field);
|
|
field->setValue(ui->enableThrustPIDScalingCheckBox->isChecked() ? "TRUE" : "FALSE");
|
|
}
|
|
|
|
void ConfigStabilizationWidget::setupExpoPlot()
|
|
{
|
|
ui->expoPlot->setMouseTracking(false);
|
|
ui->expoPlot->setAxisScale(QwtPlot::xBottom, 0.0, 1.0, 0.25);
|
|
ui->expoPlot->setAxisScale(QwtPlot::yLeft, 0.0, 1.0, 0.25);
|
|
ui->expoPlot->canvas()->setFrameShape(QFrame::NoFrame);
|
|
|
|
m_plotGrid.setMajPen(QColor(Qt::gray));
|
|
m_plotGrid.setMinPen(QColor(Qt::lightGray));
|
|
m_plotGrid.enableXMin(false);
|
|
m_plotGrid.enableYMin(false);
|
|
m_plotGrid.attach(ui->expoPlot);
|
|
|
|
m_expoPlotCurveRoll.setRenderHint(QwtPlotCurve::RenderAntialiased);
|
|
m_expoPlotCurveRoll.setPen(QPen(QColor(Qt::red).setAlpha(200), 3));
|
|
m_expoPlotCurveRoll.attach(ui->expoPlot);
|
|
replotExpoRoll(ui->expoSpinnerRoll->value());
|
|
|
|
m_expoPlotCurvePitch.setRenderHint(QwtPlotCurve::RenderAntialiased);
|
|
m_expoPlotCurvePitch.setPen(QPen(QColor(Qt::green).setAlpha(200), 3));
|
|
m_expoPlotCurvePitch.attach(ui->expoPlot);
|
|
replotExpoPitch(ui->expoSpinnerPitch->value());
|
|
|
|
m_expoPlotCurveYaw.setRenderHint(QwtPlotCurve::RenderAntialiased);
|
|
m_expoPlotCurveYaw.setPen(QPen(QColor(Qt::blue).setAlpha(200), 3));
|
|
m_expoPlotCurveYaw.attach(ui->expoPlot);
|
|
replotExpoYaw(ui->expoSpinnerYaw->value());
|
|
}
|
|
|
|
void ConfigStabilizationWidget::resetThrottleCurveToDefault()
|
|
{
|
|
UAVDataObject *defaultStabBank = (UAVDataObject *)getObjectManager()->getObject(QString(m_pidTabBars.at(0)->tabData(m_currentPIDBank).toString()));
|
|
|
|
Q_ASSERT(defaultStabBank);
|
|
defaultStabBank = defaultStabBank->dirtyClone();
|
|
|
|
UAVObjectField *field = defaultStabBank->getField("ThrustPIDScaleCurve");
|
|
Q_ASSERT(field);
|
|
|
|
QList<double> curve;
|
|
for (quint32 i = 0; i < field->getNumElements(); i++) {
|
|
curve.append(field->getValue(i).toDouble());
|
|
}
|
|
|
|
ui->thrustPIDScalingCurve->setCurve(&curve);
|
|
|
|
field = defaultStabBank->getField("EnableThrustPIDScaling");
|
|
Q_ASSERT(field);
|
|
|
|
bool enabled = field->getValue() == "TRUE";
|
|
ui->enableThrustPIDScalingCheckBox->setChecked(enabled);
|
|
ui->thrustPIDScalingCurve->setEnabled(enabled);
|
|
|
|
delete defaultStabBank;
|
|
}
|
|
|
|
void ConfigStabilizationWidget::throttleCurveUpdated()
|
|
{
|
|
setDirty(true);
|
|
}
|
|
|
|
void ConfigStabilizationWidget::replotExpo(int value, QwtPlotCurve &curve)
|
|
{
|
|
double x[EXPO_CURVE_POINTS] = { 0 };
|
|
double y[EXPO_CURVE_POINTS] = { 0 };
|
|
double factor = pow(1.03293, value);
|
|
double step = 1.0 / (EXPO_CURVE_POINTS - 1);
|
|
|
|
for (int i = 0; i < EXPO_CURVE_POINTS; i++) {
|
|
x[i] = i * step;
|
|
y[i] = pow(x[i], factor);
|
|
}
|
|
curve.setSamples(x, y, EXPO_CURVE_POINTS);
|
|
curve.show();
|
|
ui->expoPlot->replot();
|
|
}
|
|
|
|
void ConfigStabilizationWidget::replotExpoRoll(int value)
|
|
{
|
|
replotExpo(value, m_expoPlotCurveRoll);
|
|
}
|
|
|
|
void ConfigStabilizationWidget::replotExpoPitch(int value)
|
|
{
|
|
replotExpo(value, m_expoPlotCurvePitch);
|
|
}
|
|
|
|
void ConfigStabilizationWidget::replotExpoYaw(int value)
|
|
{
|
|
replotExpo(value, m_expoPlotCurveYaw);
|
|
}
|
|
|
|
void ConfigStabilizationWidget::realtimeUpdatesSlot(bool value)
|
|
{
|
|
ui->realTimeUpdates_6->setChecked(value);
|
|
ui->realTimeUpdates_8->setChecked(value);
|
|
ui->realTimeUpdates_12->setChecked(value);
|
|
ui->realTimeUpdates_7->setChecked(value);
|
|
|
|
if (value && !realtimeUpdates->isActive()) {
|
|
realtimeUpdates->start(AUTOMATIC_UPDATE_RATE);
|
|
qDebug() << "Instant Update timer started.";
|
|
} else if (!value && realtimeUpdates->isActive()) {
|
|
realtimeUpdates->stop();
|
|
qDebug() << "Instant Update timer stopped.";
|
|
}
|
|
}
|
|
|
|
void ConfigStabilizationWidget::linkCheckBoxes(bool value)
|
|
{
|
|
if (sender() == ui->checkBox_7) {
|
|
ui->checkBox_3->setChecked(value);
|
|
} else if (sender() == ui->checkBox_3) {
|
|
ui->checkBox_7->setChecked(value);
|
|
} else if (sender() == ui->checkBox_8) {
|
|
ui->checkBox_2->setChecked(value);
|
|
} else if (sender() == ui->checkBox_2) {
|
|
ui->checkBox_8->setChecked(value);
|
|
} else if (sender() == ui->basicResponsivenessCheckBox) {
|
|
ui->advancedResponsivenessCheckBox->setChecked(!value);
|
|
ui->basicResponsivenessControls->setEnabled(value);
|
|
ui->advancedResponsivenessControls->setEnabled(!value);
|
|
if (value) {
|
|
processLinkedWidgets(ui->AttitudeResponsivenessSlider);
|
|
processLinkedWidgets(ui->RateResponsivenessSlider);
|
|
}
|
|
} else if (sender() == ui->advancedResponsivenessCheckBox) {
|
|
ui->basicResponsivenessCheckBox->setChecked(!value);
|
|
ui->basicResponsivenessControls->setEnabled(!value);
|
|
ui->advancedResponsivenessControls->setEnabled(value);
|
|
}
|
|
}
|
|
|
|
void ConfigStabilizationWidget::processLinkedWidgets(QWidget *widget)
|
|
{
|
|
if (ui->checkBox_7->isChecked()) {
|
|
if (widget == ui->RateRollKp_2) {
|
|
ui->RatePitchKp->setValue(ui->RateRollKp_2->value());
|
|
} else if (widget == ui->RateRollKi_2) {
|
|
ui->RatePitchKi->setValue(ui->RateRollKi_2->value());
|
|
} else if (widget == ui->RatePitchKp) {
|
|
ui->RateRollKp_2->setValue(ui->RatePitchKp->value());
|
|
} else if (widget == ui->RatePitchKi) {
|
|
ui->RateRollKi_2->setValue(ui->RatePitchKi->value());
|
|
} else if (widget == ui->RollRateKd) {
|
|
ui->PitchRateKd->setValue(ui->RollRateKd->value());
|
|
} else if (widget == ui->PitchRateKd) {
|
|
ui->RollRateKd->setValue(ui->PitchRateKd->value());
|
|
}
|
|
}
|
|
|
|
if (ui->checkBox_8->isChecked()) {
|
|
if (widget == ui->AttitudeRollKp) {
|
|
ui->AttitudePitchKp_2->setValue(ui->AttitudeRollKp->value());
|
|
} else if (widget == ui->AttitudeRollKi) {
|
|
ui->AttitudePitchKi_2->setValue(ui->AttitudeRollKi->value());
|
|
} else if (widget == ui->AttitudePitchKp_2) {
|
|
ui->AttitudeRollKp->setValue(ui->AttitudePitchKp_2->value());
|
|
} else if (widget == ui->AttitudePitchKi_2) {
|
|
ui->AttitudeRollKi->setValue(ui->AttitudePitchKi_2->value());
|
|
}
|
|
}
|
|
|
|
if (ui->basicResponsivenessCheckBox->isChecked()) {
|
|
if (widget == ui->AttitudeResponsivenessSlider) {
|
|
ui->ratePitchKp_4->setValue(ui->AttitudeResponsivenessSlider->value());
|
|
} else if (widget == ui->RateResponsivenessSlider) {
|
|
ui->ratePitchKi_4->setValue(ui->RateResponsivenessSlider->value());
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConfigStabilizationWidget::onBoardConnected()
|
|
{
|
|
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
|
|
UAVObjectUtilManager *utilMngr = pm->getObject<UAVObjectUtilManager>();
|
|
|
|
Q_ASSERT(utilMngr);
|
|
boardModel = utilMngr->getBoardModel();
|
|
// If Revolution board enable Althold tab, otherwise disable it
|
|
ui->AltitudeHold->setEnabled((boardModel & 0xff00) == 0x0900);
|
|
}
|
|
|
|
void ConfigStabilizationWidget::pidBankChanged(int index)
|
|
{
|
|
bool dirty = isDirty();
|
|
|
|
updateObjectFromThrottleCurve();
|
|
foreach(QTabBar * tabBar, m_pidTabBars) {
|
|
disconnect(tabBar, SIGNAL(currentChanged(int)), this, SLOT(pidBankChanged(int)));
|
|
tabBar->setCurrentIndex(index);
|
|
connect(tabBar, SIGNAL(currentChanged(int)), this, SLOT(pidBankChanged(int)));
|
|
}
|
|
|
|
for (int i = 0; i < m_pidTabBars.at(0)->count(); i++) {
|
|
setWidgetBindingObjectEnabled(m_pidTabBars.at(0)->tabData(i).toString(), false);
|
|
}
|
|
|
|
setWidgetBindingObjectEnabled(m_pidTabBars.at(0)->tabData(index).toString(), true);
|
|
|
|
m_currentPIDBank = index;
|
|
qDebug() << "current bank:" << m_currentPIDBank;
|
|
updateThrottleCurveFromObject();
|
|
setDirty(dirty);
|
|
}
|
|
|
|
bool ConfigStabilizationWidget::shouldObjectBeSaved(UAVObject *object)
|
|
{
|
|
// AltitudeHoldSettings should only be saved for Revolution board to avoid error.
|
|
if ((boardModel & 0xff00) != 0x0900) {
|
|
return dynamic_cast<AltitudeHoldSettings *>(object) == 0;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
QString ConfigStabilizationWidget::mapObjectName(const QString objectName)
|
|
{
|
|
if (objectName == "StabilizationSettingsBankX") {
|
|
return m_stabilizationObjectsString;
|
|
}
|
|
return ConfigTaskWidget::mapObjectName(objectName);
|
|
}
|