mirror of
https://bitbucket.org/librepilot/librepilot.git
synced 2025-01-25 10:52:11 +01:00
1723 lines
69 KiB
C++
1723 lines
69 KiB
C++
/**
|
|
******************************************************************************
|
|
*
|
|
* @file configinputwidget.cpp
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
* @addtogroup GCSPlugins GCS Plugins
|
|
* @{
|
|
* @addtogroup ConfigPlugin Config Plugin
|
|
* @{
|
|
* @brief Servo input/output configuration panel for the config gadget
|
|
*****************************************************************************/
|
|
/*
|
|
* 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 "configinputwidget.h"
|
|
|
|
#include <extensionsystem/pluginmanager.h>
|
|
#include <coreplugin/generalsettings.h>
|
|
|
|
#include <QDebug>
|
|
#include <QStringList>
|
|
#include <QWidget>
|
|
#include <QTextEdit>
|
|
#include <QVBoxLayout>
|
|
#include <QPushButton>
|
|
#include <QDesktopServices>
|
|
#include <QUrl>
|
|
#include <QMessageBox>
|
|
#include <utils/stylehelper.h>
|
|
#include <QMessageBox>
|
|
|
|
#define ACCESS_MIN_MOVE -3
|
|
#define ACCESS_MAX_MOVE 3
|
|
#define STICK_MIN_MOVE -8
|
|
#define STICK_MAX_MOVE 8
|
|
|
|
ConfigInputWidget::ConfigInputWidget(QWidget *parent) :
|
|
ConfigTaskWidget(parent),
|
|
wizardStep(wizardNone),
|
|
// not currently stored in the settings UAVO
|
|
transmitterMode(mode2),
|
|
transmitterType(acro),
|
|
//
|
|
loop(NULL),
|
|
skipflag(false),
|
|
nextDelayedTimer(),
|
|
nextDelayedTick(0),
|
|
nextDelayedLatestActivityTick(0),
|
|
accessoryDesiredObj0(NULL),
|
|
accessoryDesiredObj1(NULL),
|
|
accessoryDesiredObj2(NULL)
|
|
{
|
|
manualCommandObj = ManualControlCommand::GetInstance(getObjectManager());
|
|
manualSettingsObj = ManualControlSettings::GetInstance(getObjectManager());
|
|
flightModeSettingsObj = FlightModeSettings::GetInstance(getObjectManager());
|
|
flightStatusObj = FlightStatus::GetInstance(getObjectManager());
|
|
receiverActivityObj = ReceiverActivity::GetInstance(getObjectManager());
|
|
accessoryDesiredObj0 = AccessoryDesired::GetInstance(getObjectManager(), 0);
|
|
accessoryDesiredObj1 = AccessoryDesired::GetInstance(getObjectManager(), 1);
|
|
accessoryDesiredObj2 = AccessoryDesired::GetInstance(getObjectManager(), 2);
|
|
actuatorSettingsObj = ActuatorSettings::GetInstance(getObjectManager());
|
|
|
|
// Only instance 0 is present if the board is not connected.
|
|
// The other instances are populated lazily.
|
|
Q_ASSERT(accessoryDesiredObj0);
|
|
|
|
ui = new Ui_InputWidget();
|
|
ui->setupUi(this);
|
|
|
|
addApplySaveButtons(ui->saveRCInputToRAM, ui->saveRCInputToSD);
|
|
|
|
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
|
|
Core::Internal::GeneralSettings *settings = pm->getObject<Core::Internal::GeneralSettings>();
|
|
if (!settings->useExpertMode()) {
|
|
ui->saveRCInputToRAM->setVisible(false);
|
|
}
|
|
|
|
addApplySaveButtons(ui->saveRCInputToRAM, ui->saveRCInputToSD);
|
|
|
|
// Generate the rows of buttons in the input channel form GUI
|
|
unsigned int index = 0;
|
|
unsigned int indexRT = 0;
|
|
foreach(QString name, manualSettingsObj->getField("ChannelNumber")->getElementNames()) {
|
|
Q_ASSERT(index < ManualControlSettings::CHANNELGROUPS_NUMELEM);
|
|
InputChannelForm *form = new InputChannelForm(index, this);
|
|
form->setName(name);
|
|
|
|
form->moveTo(*(ui->channelLayout));
|
|
|
|
// The order of the following binding calls is important. Since the values will be populated
|
|
// in reverse order of the binding order otherwise the 'Reversed' logic will floor the neutral value
|
|
// to the max value ( which is smaller than the neutral value when reversed ) and the channel number
|
|
// will not be set correctly.
|
|
addWidgetBinding("ManualControlSettings", "ChannelNumber", form->ui->channelNumber, index);
|
|
addWidgetBinding("ManualControlSettings", "ChannelGroups", form->ui->channelGroup, index);
|
|
// Slider position based on real time Rcinput (allow monitoring)
|
|
addWidgetBinding("ManualControlCommand", "Channel", form->ui->channelNeutral, index);
|
|
// Neutral value stored on board (SpinBox)
|
|
addWidgetBinding("ManualControlSettings", "ChannelNeutral", form->ui->neutralValue, index);
|
|
addWidgetBinding("ManualControlSettings", "ChannelMax", form->ui->channelMax, index);
|
|
addWidgetBinding("ManualControlSettings", "ChannelMin", form->ui->channelMin, index);
|
|
addWidgetBinding("ManualControlSettings", "ChannelMax", form->ui->channelMax, index);
|
|
|
|
addWidget(form->ui->channelRev);
|
|
|
|
// Reversing supported for some channels only
|
|
bool reversable = ((index == ManualControlSettings::CHANNELGROUPS_THROTTLE) ||
|
|
(index == ManualControlSettings::CHANNELGROUPS_ROLL) ||
|
|
(index == ManualControlSettings::CHANNELGROUPS_PITCH) ||
|
|
(index == ManualControlSettings::CHANNELGROUPS_YAW));
|
|
form->ui->channelRev->setVisible(reversable);
|
|
|
|
// Input filter response time fields supported for some channels only
|
|
switch (index) {
|
|
case ManualControlSettings::CHANNELGROUPS_ROLL:
|
|
case ManualControlSettings::CHANNELGROUPS_PITCH:
|
|
case ManualControlSettings::CHANNELGROUPS_YAW:
|
|
case ManualControlSettings::CHANNELGROUPS_COLLECTIVE:
|
|
case ManualControlSettings::CHANNELGROUPS_ACCESSORY0:
|
|
case ManualControlSettings::CHANNELGROUPS_ACCESSORY1:
|
|
case ManualControlSettings::CHANNELGROUPS_ACCESSORY2:
|
|
addWidgetBinding("ManualControlSettings", "ResponseTime", form->ui->channelResponseTime, indexRT);
|
|
++indexRT;
|
|
break;
|
|
case ManualControlSettings::CHANNELGROUPS_THROTTLE:
|
|
case ManualControlSettings::CHANNELGROUPS_FLIGHTMODE:
|
|
form->ui->channelResponseTime->setVisible(false);
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
break;
|
|
}
|
|
|
|
++index;
|
|
}
|
|
|
|
addWidgetBinding("ManualControlSettings", "Deadband", ui->deadband, 0, 0.01f);
|
|
addWidgetBinding("ManualControlSettings", "DeadbandAssistedControl", ui->assistedControlDeadband, 0, 0.01f);
|
|
|
|
connect(ui->configurationWizard, SIGNAL(clicked()), this, SLOT(goToWizard()));
|
|
connect(ui->stackedWidget, SIGNAL(currentChanged(int)), this, SLOT(disableWizardButton(int)));
|
|
connect(ui->runCalibration, SIGNAL(toggled(bool)), this, SLOT(simpleCalibration(bool)));
|
|
|
|
connect(ui->wzNext, SIGNAL(clicked()), this, SLOT(wzNext()));
|
|
connect(ui->wzCancel, SIGNAL(clicked()), this, SLOT(wzCancel()));
|
|
connect(ui->wzBack, SIGNAL(clicked()), this, SLOT(wzBack()));
|
|
|
|
ui->stackedWidget->setCurrentIndex(0);
|
|
addWidgetBinding("FlightModeSettings", "FlightModePosition", ui->fmsModePos1, 0, 1, true);
|
|
addWidgetBinding("FlightModeSettings", "FlightModePosition", ui->fmsModePos2, 1, 1, true);
|
|
addWidgetBinding("FlightModeSettings", "FlightModePosition", ui->fmsModePos3, 2, 1, true);
|
|
addWidgetBinding("FlightModeSettings", "FlightModePosition", ui->fmsModePos4, 3, 1, true);
|
|
addWidgetBinding("FlightModeSettings", "FlightModePosition", ui->fmsModePos5, 4, 1, true);
|
|
addWidgetBinding("FlightModeSettings", "FlightModePosition", ui->fmsModePos6, 5, 1, true);
|
|
addWidgetBinding("ManualControlSettings", "FlightModeNumber", ui->fmsPosNum);
|
|
|
|
addWidgetBinding("FlightModeSettings", "Stabilization1Settings", ui->fmsSsPos1Roll, "Roll", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization2Settings", ui->fmsSsPos2Roll, "Roll", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization3Settings", ui->fmsSsPos3Roll, "Roll", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization4Settings", ui->fmsSsPos4Roll, "Roll", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization5Settings", ui->fmsSsPos5Roll, "Roll", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization6Settings", ui->fmsSsPos6Roll, "Roll", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization1Settings", ui->fmsSsPos1Pitch, "Pitch", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization2Settings", ui->fmsSsPos2Pitch, "Pitch", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization3Settings", ui->fmsSsPos3Pitch, "Pitch", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization4Settings", ui->fmsSsPos4Pitch, "Pitch", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization5Settings", ui->fmsSsPos5Pitch, "Pitch", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization6Settings", ui->fmsSsPos6Pitch, "Pitch", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization1Settings", ui->fmsSsPos1Yaw, "Yaw", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization2Settings", ui->fmsSsPos2Yaw, "Yaw", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization3Settings", ui->fmsSsPos3Yaw, "Yaw", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization4Settings", ui->fmsSsPos4Yaw, "Yaw", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization5Settings", ui->fmsSsPos5Yaw, "Yaw", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization6Settings", ui->fmsSsPos6Yaw, "Yaw", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization1Settings", ui->fmsSsPos1Thrust, "Thrust", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization2Settings", ui->fmsSsPos2Thrust, "Thrust", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization3Settings", ui->fmsSsPos3Thrust, "Thrust", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization4Settings", ui->fmsSsPos4Thrust, "Thrust", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization5Settings", ui->fmsSsPos5Thrust, "Thrust", 1, true);
|
|
addWidgetBinding("FlightModeSettings", "Stabilization6Settings", ui->fmsSsPos6Thrust, "Thrust", 1, true);
|
|
|
|
addWidgetBinding("FlightModeSettings", "Arming", ui->armControl);
|
|
addWidgetBinding("FlightModeSettings", "ArmedTimeout", ui->armTimeout, 0, 1000);
|
|
connect(ManualControlCommand::GetInstance(getObjectManager()), SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveFMSlider()));
|
|
connect(ManualControlSettings::GetInstance(getObjectManager()), SIGNAL(objectUpdated(UAVObject *)), this, SLOT(updatePositionSlider()));
|
|
|
|
addWidget(ui->configurationWizard);
|
|
addWidget(ui->runCalibration);
|
|
|
|
autoLoadWidgets();
|
|
|
|
populateWidgets();
|
|
refreshWidgetsValues();
|
|
// Connect the help button
|
|
connect(ui->inputHelp, SIGNAL(clicked()), this, SLOT(openHelp()));
|
|
|
|
ui->graphicsView->setScene(new QGraphicsScene(this));
|
|
ui->graphicsView->setViewportUpdateMode(QGraphicsView::FullViewportUpdate);
|
|
m_renderer = new QSvgRenderer();
|
|
QGraphicsScene *l_scene = ui->graphicsView->scene();
|
|
ui->graphicsView->setBackgroundBrush(QBrush(Utils::StyleHelper::baseColor()));
|
|
if (QFile::exists(":/configgadget/images/TX2.svg") && m_renderer->load(QString(":/configgadget/images/TX2.svg")) && m_renderer->isValid()) {
|
|
l_scene->clear(); // Deletes all items contained in the scene as well.
|
|
|
|
m_txBackground = new QGraphicsSvgItem();
|
|
// All other items will be clipped to the shape of the background
|
|
m_txBackground->setFlags(QGraphicsItem::ItemClipsChildrenToShape |
|
|
QGraphicsItem::ItemClipsToShape);
|
|
m_txBackground->setSharedRenderer(m_renderer);
|
|
m_txBackground->setElementId("background");
|
|
l_scene->addItem(m_txBackground);
|
|
|
|
m_txMainBody = new QGraphicsSvgItem();
|
|
m_txMainBody->setParentItem(m_txBackground);
|
|
m_txMainBody->setSharedRenderer(m_renderer);
|
|
m_txMainBody->setElementId("body");
|
|
l_scene->addItem(m_txMainBody);
|
|
|
|
m_txLeftStick = new QGraphicsSvgItem();
|
|
m_txLeftStick->setParentItem(m_txBackground);
|
|
m_txLeftStick->setSharedRenderer(m_renderer);
|
|
m_txLeftStick->setElementId("ljoy");
|
|
|
|
m_txRightStick = new QGraphicsSvgItem();
|
|
m_txRightStick->setParentItem(m_txBackground);
|
|
m_txRightStick->setSharedRenderer(m_renderer);
|
|
m_txRightStick->setElementId("rjoy");
|
|
|
|
m_txAccess0 = new QGraphicsSvgItem();
|
|
m_txAccess0->setParentItem(m_txBackground);
|
|
m_txAccess0->setSharedRenderer(m_renderer);
|
|
m_txAccess0->setElementId("access0");
|
|
|
|
m_txAccess1 = new QGraphicsSvgItem();
|
|
m_txAccess1->setParentItem(m_txBackground);
|
|
m_txAccess1->setSharedRenderer(m_renderer);
|
|
m_txAccess1->setElementId("access1");
|
|
|
|
m_txAccess2 = new QGraphicsSvgItem();
|
|
m_txAccess2->setParentItem(m_txBackground);
|
|
m_txAccess2->setSharedRenderer(m_renderer);
|
|
m_txAccess2->setElementId("access2");
|
|
|
|
m_txFlightMode = new QGraphicsSvgItem();
|
|
m_txFlightMode->setParentItem(m_txBackground);
|
|
m_txFlightMode->setSharedRenderer(m_renderer);
|
|
m_txFlightMode->setElementId("flightModeCenter");
|
|
m_txFlightMode->setZValue(-10);
|
|
|
|
m_txArrows = new QGraphicsSvgItem();
|
|
m_txArrows->setParentItem(m_txBackground);
|
|
m_txArrows->setSharedRenderer(m_renderer);
|
|
m_txArrows->setElementId("arrows");
|
|
m_txArrows->setVisible(false);
|
|
|
|
QRectF orig = m_renderer->boundsOnElement("ljoy");
|
|
QMatrix Matrix = m_renderer->matrixForElement("ljoy");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txLeftStickOrig.translate(orig.x(), orig.y());
|
|
m_txLeftStick->setTransform(m_txLeftStickOrig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("arrows");
|
|
Matrix = m_renderer->matrixForElement("arrows");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txArrowsOrig.translate(orig.x(), orig.y());
|
|
m_txArrows->setTransform(m_txArrowsOrig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("body");
|
|
Matrix = m_renderer->matrixForElement("body");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txMainBodyOrig.translate(orig.x(), orig.y());
|
|
m_txMainBody->setTransform(m_txMainBodyOrig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("flightModeCenter");
|
|
Matrix = m_renderer->matrixForElement("flightModeCenter");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txFlightModeCOrig.translate(orig.x(), orig.y());
|
|
m_txFlightMode->setTransform(m_txFlightModeCOrig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("flightModeLeft");
|
|
Matrix = m_renderer->matrixForElement("flightModeLeft");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txFlightModeLOrig.translate(orig.x(), orig.y());
|
|
orig = m_renderer->boundsOnElement("flightModeRight");
|
|
Matrix = m_renderer->matrixForElement("flightModeRight");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txFlightModeROrig.translate(orig.x(), orig.y());
|
|
|
|
orig = m_renderer->boundsOnElement("rjoy");
|
|
Matrix = m_renderer->matrixForElement("rjoy");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txRightStickOrig.translate(orig.x(), orig.y());
|
|
m_txRightStick->setTransform(m_txRightStickOrig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("access0");
|
|
Matrix = m_renderer->matrixForElement("access0");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txAccess0Orig.translate(orig.x(), orig.y());
|
|
m_txAccess0->setTransform(m_txAccess0Orig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("access1");
|
|
Matrix = m_renderer->matrixForElement("access1");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txAccess1Orig.translate(orig.x(), orig.y());
|
|
m_txAccess1->setTransform(m_txAccess1Orig, false);
|
|
|
|
orig = m_renderer->boundsOnElement("access2");
|
|
Matrix = m_renderer->matrixForElement("access2");
|
|
orig = Matrix.mapRect(orig);
|
|
m_txAccess2Orig.translate(orig.x(), orig.y());
|
|
m_txAccess2->setTransform(m_txAccess2Orig, true);
|
|
}
|
|
ui->graphicsView->fitInView(m_txMainBody, Qt::KeepAspectRatio);
|
|
animate = new QTimer(this);
|
|
connect(animate, SIGNAL(timeout()), this, SLOT(moveTxControls()));
|
|
|
|
heliChannelOrder << ManualControlSettings::CHANNELGROUPS_COLLECTIVE <<
|
|
ManualControlSettings::CHANNELGROUPS_THROTTLE <<
|
|
ManualControlSettings::CHANNELGROUPS_ROLL <<
|
|
ManualControlSettings::CHANNELGROUPS_PITCH <<
|
|
ManualControlSettings::CHANNELGROUPS_YAW <<
|
|
ManualControlSettings::CHANNELGROUPS_FLIGHTMODE <<
|
|
ManualControlSettings::CHANNELGROUPS_ACCESSORY0 <<
|
|
ManualControlSettings::CHANNELGROUPS_ACCESSORY1 <<
|
|
ManualControlSettings::CHANNELGROUPS_ACCESSORY2;
|
|
|
|
acroChannelOrder << ManualControlSettings::CHANNELGROUPS_THROTTLE <<
|
|
ManualControlSettings::CHANNELGROUPS_ROLL <<
|
|
ManualControlSettings::CHANNELGROUPS_PITCH <<
|
|
ManualControlSettings::CHANNELGROUPS_YAW <<
|
|
ManualControlSettings::CHANNELGROUPS_FLIGHTMODE <<
|
|
ManualControlSettings::CHANNELGROUPS_ACCESSORY0 <<
|
|
ManualControlSettings::CHANNELGROUPS_ACCESSORY1 <<
|
|
ManualControlSettings::CHANNELGROUPS_ACCESSORY2;
|
|
|
|
updateEnableControls();
|
|
}
|
|
|
|
void ConfigInputWidget::resetTxControls()
|
|
{
|
|
m_txLeftStick->setTransform(m_txLeftStickOrig, false);
|
|
m_txRightStick->setTransform(m_txRightStickOrig, false);
|
|
m_txAccess0->setTransform(m_txAccess0Orig, false);
|
|
m_txAccess1->setTransform(m_txAccess1Orig, false);
|
|
m_txAccess2->setTransform(m_txAccess2Orig, false);
|
|
m_txFlightMode->setElementId("flightModeCenter");
|
|
m_txFlightMode->setTransform(m_txFlightModeCOrig, false);
|
|
m_txArrows->setVisible(false);
|
|
}
|
|
|
|
ConfigInputWidget::~ConfigInputWidget()
|
|
{}
|
|
|
|
void ConfigInputWidget::enableControls(bool enable)
|
|
{
|
|
ConfigTaskWidget::enableControls(enable);
|
|
|
|
if (enable) {
|
|
updatePositionSlider();
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::resizeEvent(QResizeEvent *event)
|
|
{
|
|
QWidget::resizeEvent(event);
|
|
|
|
ui->graphicsView->fitInView(m_txBackground, Qt::KeepAspectRatio);
|
|
}
|
|
|
|
void ConfigInputWidget::openHelp()
|
|
{
|
|
QDesktopServices::openUrl(QUrl(tr("http://wiki.openpilot.org/x/04Cf"), QUrl::StrictMode));
|
|
}
|
|
|
|
void ConfigInputWidget::goToWizard()
|
|
{
|
|
QMessageBox msgBox;
|
|
|
|
msgBox.setText(tr("Arming Settings are now set to 'Always Disarmed' for your safety."));
|
|
msgBox.setDetailedText(tr("You will have to reconfigure the arming settings manually "
|
|
"when the wizard is finished. After the last step of the "
|
|
"wizard you will be taken to the Arming Settings screen."));
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.setDefaultButton(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
|
|
// Set correct tab visible before starting wizard.
|
|
if (ui->tabWidget->currentIndex() != 0) {
|
|
ui->tabWidget->setCurrentIndex(0);
|
|
}
|
|
|
|
// Stash current manual settings data in case the wizard is
|
|
// cancelled or the user proceeds far enough into the wizard such
|
|
// that the UAVO is changed, but then backs out to the start and
|
|
// chooses a different TX type (which could otherwise result in
|
|
// unexpected TX channels being enabled)
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
previousManualSettingsData = manualSettingsData;
|
|
flightModeSettingsData = flightModeSettingsObj->getData();
|
|
previousFlightModeSettingsData = flightModeSettingsData;
|
|
flightModeSettingsData.Arming = FlightModeSettings::ARMING_ALWAYSDISARMED;
|
|
flightModeSettingsObj->setData(flightModeSettingsData);
|
|
// Stash actuatorSettings
|
|
actuatorSettingsData = actuatorSettingsObj->getData();
|
|
previousActuatorSettingsData = actuatorSettingsData;
|
|
|
|
// Now reset channel and actuator settings (disable outputs)
|
|
resetChannelSettings();
|
|
resetActuatorSettings();
|
|
|
|
// Use faster input update rate.
|
|
fastMdata();
|
|
|
|
// start the wizard
|
|
wizardSetUpStep(wizardWelcome);
|
|
ui->graphicsView->fitInView(m_txBackground, Qt::KeepAspectRatio);
|
|
}
|
|
|
|
void ConfigInputWidget::disableWizardButton(int value)
|
|
{
|
|
if (value != 0) {
|
|
ui->groupBox_3->setVisible(false);
|
|
} else {
|
|
ui->groupBox_3->setVisible(true);
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::wzCancel()
|
|
{
|
|
dimOtherControls(false);
|
|
|
|
// Cancel any ongoing delayd next trigger.
|
|
wzNextDelayedCancel();
|
|
|
|
// Restore original input update rate.
|
|
restoreMdata();
|
|
|
|
ui->stackedWidget->setCurrentIndex(0);
|
|
|
|
if (wizardStep != wizardNone) {
|
|
wizardTearDownStep(wizardStep);
|
|
}
|
|
wizardStep = wizardNone;
|
|
ui->stackedWidget->setCurrentIndex(0);
|
|
|
|
// Load settings back from beginning of wizard
|
|
manualSettingsObj->setData(previousManualSettingsData);
|
|
flightModeSettingsObj->setData(previousFlightModeSettingsData);
|
|
actuatorSettingsObj->setData(previousActuatorSettingsData);
|
|
}
|
|
|
|
void ConfigInputWidget::registerControlActivity()
|
|
{
|
|
nextDelayedLatestActivityTick = nextDelayedTick;
|
|
}
|
|
|
|
void ConfigInputWidget::wzNextDelayed()
|
|
{
|
|
nextDelayedTick++;
|
|
|
|
// Call next after the full 2500 ms timeout has been reached,
|
|
// or if no input activity has occurred the last 500 ms.
|
|
if (nextDelayedTick == 25 ||
|
|
nextDelayedTick - nextDelayedLatestActivityTick >= 5) {
|
|
wzNext();
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::wzNextDelayedStart()
|
|
{
|
|
// Call wzNextDelayed every 100 ms, to see if it's time to go to the next page.
|
|
connect(&nextDelayedTimer, SIGNAL(timeout()), this, SLOT(wzNextDelayed()));
|
|
nextDelayedTimer.start(100);
|
|
}
|
|
|
|
// Cancel the delayed next timer, if it's active.
|
|
void ConfigInputWidget::wzNextDelayedCancel()
|
|
{
|
|
nextDelayedTick = 0;
|
|
nextDelayedLatestActivityTick = 0;
|
|
if (nextDelayedTimer.isActive()) {
|
|
nextDelayedTimer.stop();
|
|
disconnect(&nextDelayedTimer, SIGNAL(timeout()), this, SLOT(wzNextDelayed()));
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::wzNext()
|
|
{
|
|
wzNextDelayedCancel();
|
|
|
|
// In identify sticks mode the next button can indicate
|
|
// channel advance
|
|
if (wizardStep != wizardNone &&
|
|
wizardStep != wizardIdentifySticks) {
|
|
wizardTearDownStep(wizardStep);
|
|
}
|
|
|
|
// State transitions for next button
|
|
switch (wizardStep) {
|
|
case wizardWelcome:
|
|
wizardSetUpStep(wizardChooseType);
|
|
break;
|
|
case wizardChooseType:
|
|
wizardSetUpStep(wizardChooseMode);
|
|
break;
|
|
case wizardChooseMode:
|
|
wizardSetUpStep(wizardIdentifySticks);
|
|
break;
|
|
case wizardIdentifySticks:
|
|
nextChannel();
|
|
if (currentChannelNum == -1) { // Gone through all channels
|
|
wizardTearDownStep(wizardIdentifySticks);
|
|
wizardSetUpStep(wizardIdentifyCenter);
|
|
}
|
|
break;
|
|
case wizardIdentifyCenter:
|
|
wizardSetUpStep(wizardIdentifyLimits);
|
|
break;
|
|
case wizardIdentifyLimits:
|
|
wizardSetUpStep(wizardIdentifyInverted);
|
|
break;
|
|
case wizardIdentifyInverted:
|
|
wizardSetUpStep(wizardFinish);
|
|
break;
|
|
case wizardFinish:
|
|
wizardStep = wizardNone;
|
|
|
|
// Restore original input update rate.
|
|
restoreMdata();
|
|
|
|
// Load actuator settings back from beginning of wizard
|
|
actuatorSettingsObj->setData(previousActuatorSettingsData);
|
|
|
|
// Leave setting the throttle neutral until the final Next press,
|
|
// else the throttle scaling causes the graphical stick movement to not
|
|
// match the tx stick
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_THROTTLE] =
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE] +
|
|
((manualSettingsData.ChannelMax[ManualControlSettings::CHANNELMAX_THROTTLE] -
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE]) * 0.04);
|
|
if ((abs(manualSettingsData.ChannelMax[ManualControlSettings::CHANNELMAX_FLIGHTMODE] -
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_FLIGHTMODE]) < 100) ||
|
|
(abs(manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_FLIGHTMODE] -
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_FLIGHTMODE]) < 100)) {
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_FLIGHTMODE] =
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_FLIGHTMODE] +
|
|
(manualSettingsData.ChannelMax[ManualControlSettings::CHANNELMAX_FLIGHTMODE] -
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_FLIGHTMODE]) / 2;
|
|
}
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
// move to Arming Settings tab
|
|
ui->stackedWidget->setCurrentIndex(0);
|
|
ui->tabWidget->setCurrentIndex(2);
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::wzBack()
|
|
{
|
|
wzNextDelayedCancel();
|
|
|
|
if (wizardStep != wizardNone &&
|
|
wizardStep != wizardIdentifySticks) {
|
|
wizardTearDownStep(wizardStep);
|
|
}
|
|
|
|
// State transitions for next button
|
|
switch (wizardStep) {
|
|
case wizardChooseType:
|
|
wizardSetUpStep(wizardWelcome);
|
|
break;
|
|
case wizardChooseMode:
|
|
wizardSetUpStep(wizardChooseType);
|
|
break;
|
|
case wizardIdentifySticks:
|
|
prevChannel();
|
|
if (currentChannelNum == -1) {
|
|
wizardTearDownStep(wizardIdentifySticks);
|
|
wizardSetUpStep(wizardChooseMode);
|
|
}
|
|
break;
|
|
case wizardIdentifyCenter:
|
|
wizardSetUpStep(wizardIdentifySticks);
|
|
break;
|
|
case wizardIdentifyLimits:
|
|
wizardSetUpStep(wizardIdentifyCenter);
|
|
break;
|
|
case wizardIdentifyInverted:
|
|
wizardSetUpStep(wizardIdentifyLimits);
|
|
break;
|
|
case wizardFinish:
|
|
wizardSetUpStep(wizardIdentifyInverted);
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::wizardSetUpStep(enum wizardSteps step)
|
|
{
|
|
ui->wzText2->clear();
|
|
ui->wzNext->setText(tr("Next"));
|
|
|
|
switch (step) {
|
|
case wizardWelcome:
|
|
foreach(QPointer<QWidget> wd, extraWidgets) {
|
|
if (!wd.isNull()) {
|
|
delete wd;
|
|
}
|
|
}
|
|
extraWidgets.clear();
|
|
ui->graphicsView->setVisible(false);
|
|
setTxMovement(nothing);
|
|
ui->wzText->setText(tr("Welcome to the inputs configuration wizard.\n\n"
|
|
"Please follow the instructions on the screen and only move your controls when asked to.\n"
|
|
"Make sure you already configured your hardware settings on the proper tab and restarted your board.\n\n"
|
|
"You can press 'back' at any time to return to the previous screen or press 'Cancel' to quit the wizard.\n"));
|
|
ui->stackedWidget->setCurrentIndex(1);
|
|
ui->wzBack->setEnabled(false);
|
|
break;
|
|
case wizardChooseType:
|
|
{
|
|
ui->graphicsView->setVisible(true);
|
|
ui->graphicsView->fitInView(m_txBackground, Qt::KeepAspectRatio);
|
|
setTxMovement(nothing);
|
|
ui->wzText->setText(tr("Please choose your transmitter type:"));
|
|
ui->wzBack->setEnabled(true);
|
|
QRadioButton *typeAcro = new QRadioButton(tr("Acro: normal transmitter for fixed-wing or quad"), this);
|
|
QRadioButton *typeHeli = new QRadioButton(tr("Helicopter: has collective pitch and throttle input"), this);
|
|
if (transmitterType == heli) {
|
|
typeHeli->setChecked(true);
|
|
} else {
|
|
typeAcro->setChecked(true);
|
|
}
|
|
ui->wzText2->setText(tr("If selecting the Helicopter option, please engage throttle hold now."));
|
|
|
|
extraWidgets.clear();
|
|
extraWidgets.append(typeAcro);
|
|
extraWidgets.append(typeHeli);
|
|
ui->radioButtonsLayout->layout()->addWidget(typeAcro);
|
|
ui->radioButtonsLayout->layout()->addWidget(typeHeli);
|
|
}
|
|
break;
|
|
case wizardChooseMode:
|
|
{
|
|
ui->wzBack->setEnabled(true);
|
|
extraWidgets.clear();
|
|
ui->wzText->setText(tr("Please choose your transmitter mode:"));
|
|
for (int i = 0; i <= mode4; ++i) {
|
|
QString label;
|
|
txMode mode = static_cast<txMode>(i);
|
|
if (transmitterType == heli) {
|
|
switch (mode) {
|
|
case mode1: label = tr("Mode 1: Fore/Aft Cyclic and Yaw on the left, Throttle/Collective and Left/Right Cyclic on the right"); break;
|
|
case mode2: label = tr("Mode 2: Throttle/Collective and Yaw on the left, Cyclic on the right"); break;
|
|
case mode3: label = tr("Mode 3: Cyclic on the left, Throttle/Collective and Yaw on the right"); break;
|
|
case mode4: label = tr("Mode 4: Throttle/Collective and Left/Right Cyclic on the left, Fore/Aft Cyclic and Yaw on the right"); break;
|
|
default: Q_ASSERT(0); break;
|
|
}
|
|
} else {
|
|
switch (mode) {
|
|
case mode1: label = tr("Mode 1: Elevator and Rudder on the left, Throttle and Ailerons on the right"); break;
|
|
case mode2: label = tr("Mode 2: Throttle and Rudder on the left, Elevator and Ailerons on the right"); break;
|
|
case mode3: label = tr("Mode 3: Elevator and Ailerons on the left, Throttle and Rudder on the right"); break;
|
|
case mode4: label = tr("Mode 4: Throttle and Ailerons on the left, Elevator and Rudder on the right"); break;
|
|
default: Q_ASSERT(0); break;
|
|
}
|
|
ui->wzText2->setText(tr("For a Quad: Elevator is Pitch, Ailerons are Roll, and Rudder is Yaw."));
|
|
}
|
|
QRadioButton *modeButton = new QRadioButton(label, this);
|
|
if (transmitterMode == mode) {
|
|
modeButton->setChecked(true);
|
|
}
|
|
extraWidgets.append(modeButton);
|
|
ui->radioButtonsLayout->layout()->addWidget(modeButton);
|
|
}
|
|
}
|
|
break;
|
|
case wizardIdentifySticks:
|
|
usedChannels.clear();
|
|
currentChannelNum = -1;
|
|
nextChannel();
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
connect(receiverActivityObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(identifyControls()));
|
|
ui->wzNext->setEnabled(false);
|
|
break;
|
|
case wizardIdentifyCenter:
|
|
setTxMovement(centerAll);
|
|
ui->wzText->setText(QString(tr("Please center all controls and trims and press Next when ready.\n\n"
|
|
"If your FlightMode switch has only two positions, leave it in either position.")));
|
|
break;
|
|
case wizardIdentifyLimits:
|
|
{
|
|
setTxMovement(nothing);
|
|
ui->wzText->setText(QString(tr("Please move all controls to their maximum extents on both directions.\n\nPress Next when ready.")));
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
for (uint i = 0; i < ManualControlSettings::CHANNELMAX_NUMELEM; ++i) {
|
|
// Preserve the inverted status
|
|
if (manualSettingsData.ChannelMin[i] <= manualSettingsData.ChannelMax[i]) {
|
|
manualSettingsData.ChannelMin[i] = manualSettingsData.ChannelNeutral[i];
|
|
manualSettingsData.ChannelMax[i] = manualSettingsData.ChannelNeutral[i];
|
|
} else {
|
|
// Make this detect as still inverted
|
|
manualSettingsData.ChannelMin[i] = manualSettingsData.ChannelNeutral[i] + 1;
|
|
manualSettingsData.ChannelMax[i] = manualSettingsData.ChannelNeutral[i];
|
|
}
|
|
}
|
|
connect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(identifyLimits()));
|
|
connect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
connect(flightStatusObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
connect(accessoryDesiredObj0, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
}
|
|
break;
|
|
case wizardIdentifyInverted:
|
|
dimOtherControls(true);
|
|
setTxMovement(nothing);
|
|
extraWidgets.clear();
|
|
for (int index = 0; index < manualSettingsObj->getField("ChannelMax")->getElementNames().length(); index++) {
|
|
QString name = manualSettingsObj->getField("ChannelMax")->getElementNames().at(index);
|
|
if (!name.contains("Access") && !name.contains("Flight") &&
|
|
(!name.contains("Collective") || transmitterType == heli)) {
|
|
QCheckBox *cb = new QCheckBox(name, this);
|
|
// Make sure checked status matches current one
|
|
cb->setChecked(manualSettingsData.ChannelMax[index] < manualSettingsData.ChannelMin[index]);
|
|
dynamic_cast<QGridLayout *>(ui->checkBoxesLayout->layout())->addWidget(cb, extraWidgets.size() / 4, extraWidgets.size() % 4);
|
|
extraWidgets.append(cb);
|
|
connect(cb, SIGNAL(toggled(bool)), this, SLOT(invertControls()));
|
|
}
|
|
}
|
|
connect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
ui->wzText->setText(QString(tr("Please check the picture below and correct all the sticks which show an inverted movement. Press Next when ready.")));
|
|
break;
|
|
case wizardFinish:
|
|
dimOtherControls(false);
|
|
connect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
connect(flightStatusObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
connect(accessoryDesiredObj0, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
ui->wzText->setText(QString(tr("You have completed this wizard, please check below if the picture mimics your sticks movement.\n\n"
|
|
"IMPORTANT: These new settings have not been saved to the board yet. After pressing Next you will go to the Arming Settings "
|
|
"tab where you can set your desired arming sequence and save the configuration.")));
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
}
|
|
wizardStep = step;
|
|
}
|
|
|
|
void ConfigInputWidget::wizardTearDownStep(enum wizardSteps step)
|
|
{
|
|
QRadioButton *mode, *type;
|
|
|
|
Q_ASSERT(step == wizardStep);
|
|
switch (step) {
|
|
case wizardWelcome:
|
|
break;
|
|
case wizardChooseType:
|
|
type = qobject_cast<QRadioButton *>(extraWidgets.at(0));
|
|
if (type->isChecked()) {
|
|
transmitterType = acro;
|
|
} else {
|
|
transmitterType = heli;
|
|
}
|
|
delete extraWidgets.at(0);
|
|
delete extraWidgets.at(1);
|
|
extraWidgets.clear();
|
|
break;
|
|
case wizardChooseMode:
|
|
for (int i = mode1; i <= mode4; ++i) {
|
|
mode = qobject_cast<QRadioButton *>(extraWidgets.first());
|
|
if (mode->isChecked()) {
|
|
transmitterMode = static_cast<txMode>(i);
|
|
}
|
|
delete mode;
|
|
extraWidgets.removeFirst();
|
|
}
|
|
break;
|
|
case wizardIdentifySticks:
|
|
disconnect(receiverActivityObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(identifyControls()));
|
|
ui->wzNext->setEnabled(true);
|
|
setTxMovement(nothing);
|
|
break;
|
|
case wizardIdentifyCenter:
|
|
manualCommandData = manualCommandObj->getData();
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
for (unsigned int i = 0; i < ManualControlCommand::CHANNEL_NUMELEM; ++i) {
|
|
manualSettingsData.ChannelNeutral[i] = manualCommandData.Channel[i];
|
|
}
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
setTxMovement(nothing);
|
|
break;
|
|
case wizardIdentifyLimits:
|
|
disconnect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(identifyLimits()));
|
|
disconnect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
disconnect(flightStatusObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
disconnect(accessoryDesiredObj0, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
setTxMovement(nothing);
|
|
break;
|
|
case wizardIdentifyInverted:
|
|
dimOtherControls(false);
|
|
foreach(QWidget * wd, extraWidgets) {
|
|
QCheckBox *cb = qobject_cast<QCheckBox *>(wd);
|
|
|
|
if (cb) {
|
|
disconnect(cb, SIGNAL(toggled(bool)), this, SLOT(invertControls()));
|
|
delete cb;
|
|
}
|
|
}
|
|
extraWidgets.clear();
|
|
disconnect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
break;
|
|
case wizardFinish:
|
|
dimOtherControls(false);
|
|
setTxMovement(nothing);
|
|
disconnect(manualCommandObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
disconnect(flightStatusObj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
disconnect(accessoryDesiredObj0, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(moveSticks()));
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
static void fastMdataSingle(UAVDataObject *object, UAVObject::Metadata *savedMdata)
|
|
{
|
|
*savedMdata = object->getMetadata();
|
|
UAVObject::Metadata mdata = *savedMdata;
|
|
UAVObject::SetFlightTelemetryUpdateMode(mdata, UAVObject::UPDATEMODE_PERIODIC);
|
|
mdata.flightTelemetryUpdatePeriod = 150;
|
|
object->setMetadata(mdata);
|
|
}
|
|
|
|
static void restoreMdataSingle(UAVDataObject *object, UAVObject::Metadata *savedMdata)
|
|
{
|
|
object->setMetadata(*savedMdata);
|
|
}
|
|
|
|
/**
|
|
* Set manual control command to fast updates
|
|
*/
|
|
void ConfigInputWidget::fastMdata()
|
|
{
|
|
fastMdataSingle(manualCommandObj, &manualControlMdata);
|
|
fastMdataSingle(accessoryDesiredObj0, &accessoryDesiredMdata0);
|
|
}
|
|
|
|
/**
|
|
* Restore previous update settings for manual control data
|
|
*/
|
|
void ConfigInputWidget::restoreMdata()
|
|
{
|
|
restoreMdataSingle(manualCommandObj, &manualControlMdata);
|
|
restoreMdataSingle(accessoryDesiredObj0, &accessoryDesiredMdata0);
|
|
}
|
|
|
|
/**
|
|
* Set the display to indicate which channel the person should move
|
|
*/
|
|
void ConfigInputWidget::setChannel(int newChan)
|
|
{
|
|
if (newChan == ManualControlSettings::CHANNELGROUPS_COLLECTIVE) {
|
|
ui->wzText->setText(QString(tr("Please enable throttle hold mode.\n\nMove the Collective Pitch stick.")));
|
|
} else if (newChan == ManualControlSettings::CHANNELGROUPS_FLIGHTMODE) {
|
|
ui->wzText->setText(QString(tr("Please toggle the Flight Mode switch.\n\nFor switches you may have to repeat this rapidly.")));
|
|
} else if ((transmitterType == heli) && (newChan == ManualControlSettings::CHANNELGROUPS_THROTTLE)) {
|
|
ui->wzText->setText(QString(tr("Please disable throttle hold mode.\n\nMove the Throttle stick.")));
|
|
} else {
|
|
ui->wzText->setText(QString(tr("Please move each control one at a time according to the instructions and picture below.\n\n"
|
|
"Move the %1 stick.")).arg(manualSettingsObj->getField("ChannelGroups")->getElementNames().at(newChan)));
|
|
}
|
|
|
|
if (manualSettingsObj->getField("ChannelGroups")->getElementNames().at(newChan).contains("Accessory")) {
|
|
ui->wzNext->setEnabled(true);
|
|
ui->wzNext->setText(tr("Next / Skip"));
|
|
ui->wzText->setText(ui->wzText->text() + tr(" Alternatively, click Next to skip this channel."));
|
|
} else {
|
|
ui->wzNext->setEnabled(false);
|
|
}
|
|
|
|
setMoveFromCommand(newChan);
|
|
currentChannelNum = newChan;
|
|
channelDetected = false;
|
|
}
|
|
|
|
/**
|
|
* Unfortunately order of channel should be different in different conditions. Selects
|
|
* next channel based on heli or acro mode
|
|
*/
|
|
void ConfigInputWidget::nextChannel()
|
|
{
|
|
QList <int> order = (transmitterType == heli) ? heliChannelOrder : acroChannelOrder;
|
|
|
|
if (currentChannelNum == -1) {
|
|
setChannel(order[0]);
|
|
return;
|
|
}
|
|
for (int i = 0; i < order.length() - 1; i++) {
|
|
if (order[i] == currentChannelNum) {
|
|
setChannel(order[i + 1]);
|
|
return;
|
|
}
|
|
}
|
|
currentChannelNum = -1; // hit end of list
|
|
}
|
|
|
|
/**
|
|
* Unfortunately order of channel should be different in different conditions. Selects
|
|
* previous channel based on heli or acro mode
|
|
*/
|
|
void ConfigInputWidget::prevChannel()
|
|
{
|
|
QList <int> order = transmitterType == heli ? heliChannelOrder : acroChannelOrder;
|
|
|
|
// No previous from unset channel or next state
|
|
if (currentChannelNum == -1) {
|
|
return;
|
|
}
|
|
|
|
for (int i = 1; i < order.length(); i++) {
|
|
if (order[i] == currentChannelNum) {
|
|
if (!usedChannels.isEmpty() &&
|
|
usedChannels.back().channelIndex == order[i - 1]) {
|
|
usedChannels.removeLast();
|
|
}
|
|
setChannel(order[i - 1]);
|
|
return;
|
|
}
|
|
}
|
|
currentChannelNum = -1; // hit end of list
|
|
}
|
|
|
|
void ConfigInputWidget::identifyControls()
|
|
{
|
|
static const int DEBOUNCE_COUNT = 4;
|
|
static int debounce = 0;
|
|
|
|
receiverActivityData = receiverActivityObj->getData();
|
|
|
|
if (receiverActivityData.ActiveChannel == 255) {
|
|
return;
|
|
}
|
|
|
|
if (channelDetected) {
|
|
registerControlActivity();
|
|
return;
|
|
}
|
|
|
|
receiverActivityData = receiverActivityObj->getData();
|
|
currentChannel.group = receiverActivityData.ActiveGroup;
|
|
currentChannel.number = receiverActivityData.ActiveChannel;
|
|
|
|
if (debounce == 0) {
|
|
// Register a channel to be debounced.
|
|
lastChannel.group = currentChannel.group;
|
|
lastChannel.number = currentChannel.number;
|
|
lastChannel.channelIndex = currentChannelNum;
|
|
++debounce;
|
|
return;
|
|
}
|
|
|
|
if (currentChannel != lastChannel) {
|
|
// A new channel was seen. Only register it if we count down to 0.
|
|
--debounce;
|
|
return;
|
|
}
|
|
|
|
if (debounce < DEBOUNCE_COUNT) {
|
|
// We still haven't seen enough enough activity on this channel yet.
|
|
++debounce;
|
|
return;
|
|
}
|
|
|
|
// Channel has been debounced and it's enough record it.
|
|
|
|
if (usedChannels.contains(lastChannel)) {
|
|
// Channel is already recorded.
|
|
return;
|
|
}
|
|
|
|
// Record the channel.
|
|
|
|
channelDetected = true;
|
|
debounce = 0;
|
|
usedChannels.append(lastChannel);
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
manualSettingsData.ChannelGroups[currentChannelNum] = currentChannel.group;
|
|
manualSettingsData.ChannelNumber[currentChannelNum] = currentChannel.number;
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
|
|
// m_config->wzText->clear();
|
|
setTxMovement(nothing);
|
|
|
|
wzNextDelayedStart();
|
|
}
|
|
|
|
void ConfigInputWidget::identifyLimits()
|
|
{
|
|
manualCommandData = manualCommandObj->getData();
|
|
for (uint i = 0; i < ManualControlSettings::CHANNELMAX_NUMELEM; ++i) {
|
|
if (manualSettingsData.ChannelMin[i] <= manualSettingsData.ChannelMax[i]) {
|
|
// Non inverted channel
|
|
if (manualSettingsData.ChannelMin[i] > manualCommandData.Channel[i]) {
|
|
manualSettingsData.ChannelMin[i] = manualCommandData.Channel[i];
|
|
}
|
|
if (manualSettingsData.ChannelMax[i] < manualCommandData.Channel[i]) {
|
|
manualSettingsData.ChannelMax[i] = manualCommandData.Channel[i];
|
|
}
|
|
} else {
|
|
// Inverted channel
|
|
if (manualSettingsData.ChannelMax[i] > manualCommandData.Channel[i]) {
|
|
manualSettingsData.ChannelMax[i] = manualCommandData.Channel[i];
|
|
}
|
|
if (manualSettingsData.ChannelMin[i] < manualCommandData.Channel[i]) {
|
|
manualSettingsData.ChannelMin[i] = manualCommandData.Channel[i];
|
|
}
|
|
}
|
|
}
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
}
|
|
void ConfigInputWidget::setMoveFromCommand(int command)
|
|
{
|
|
// ManualControlSettings::ChannelNumberElem:
|
|
// CHANNELNUMBER_ROLL=0,
|
|
// CHANNELNUMBER_PITCH=1,
|
|
// CHANNELNUMBER_YAW=2,
|
|
// CHANNELNUMBER_THROTTLE=3,
|
|
// CHANNELNUMBER_FLIGHTMODE=4,
|
|
// CHANNELNUMBER_ACCESSORY0=5,
|
|
// CHANNELNUMBER_ACCESSORY1=6,
|
|
// CHANNELNUMBER_ACCESSORY2=7
|
|
|
|
txMovements movement = moveLeftVerticalStick;
|
|
|
|
switch (command) {
|
|
case ManualControlSettings::CHANNELNUMBER_ROLL:
|
|
movement = ((transmitterMode == mode3 || transmitterMode == mode4) ?
|
|
moveLeftHorizontalStick : moveRightHorizontalStick);
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_PITCH:
|
|
movement = (transmitterMode == mode1 || transmitterMode == mode3) ?
|
|
moveLeftVerticalStick : moveRightVerticalStick;
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_YAW:
|
|
movement = ((transmitterMode == mode1 || transmitterMode == mode2) ?
|
|
moveLeftHorizontalStick : moveRightHorizontalStick);
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_THROTTLE:
|
|
movement = (transmitterMode == mode2 || transmitterMode == mode4) ?
|
|
moveLeftVerticalStick : moveRightVerticalStick;
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_COLLECTIVE:
|
|
movement = (transmitterMode == mode2 || transmitterMode == mode4) ?
|
|
moveLeftVerticalStick : moveRightVerticalStick;
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_FLIGHTMODE:
|
|
movement = moveFlightMode;
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_ACCESSORY0:
|
|
movement = moveAccess0;
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_ACCESSORY1:
|
|
movement = moveAccess1;
|
|
break;
|
|
case ManualControlSettings::CHANNELNUMBER_ACCESSORY2:
|
|
movement = moveAccess2;
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
break;
|
|
}
|
|
setTxMovement(movement);
|
|
}
|
|
|
|
void ConfigInputWidget::setTxMovement(txMovements movement)
|
|
{
|
|
resetTxControls();
|
|
switch (movement) {
|
|
case moveLeftVerticalStick:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveLeftVerticalStick;
|
|
animate->start(100);
|
|
break;
|
|
case moveRightVerticalStick:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveRightVerticalStick;
|
|
animate->start(100);
|
|
break;
|
|
case moveLeftHorizontalStick:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveLeftHorizontalStick;
|
|
animate->start(100);
|
|
break;
|
|
case moveRightHorizontalStick:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveRightHorizontalStick;
|
|
animate->start(100);
|
|
break;
|
|
case moveAccess0:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveAccess0;
|
|
animate->start(100);
|
|
break;
|
|
case moveAccess1:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveAccess1;
|
|
animate->start(100);
|
|
break;
|
|
case moveAccess2:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveAccess2;
|
|
animate->start(100);
|
|
break;
|
|
case moveFlightMode:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveFlightMode;
|
|
animate->start(1000);
|
|
break;
|
|
case centerAll:
|
|
movePos = 0;
|
|
currentMovement = centerAll;
|
|
animate->start(1000);
|
|
break;
|
|
case moveAll:
|
|
movePos = 0;
|
|
growing = true;
|
|
currentMovement = moveAll;
|
|
animate->start(50);
|
|
break;
|
|
case nothing:
|
|
movePos = 0;
|
|
animate->stop();
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::moveTxControls()
|
|
{
|
|
QTransform trans;
|
|
QGraphicsItem *item = NULL;
|
|
txMovementType move = vertical;
|
|
int limitMax = 0;
|
|
int limitMin = 0;
|
|
static bool auxFlag = false;
|
|
|
|
switch (currentMovement) {
|
|
case moveLeftVerticalStick:
|
|
item = m_txLeftStick;
|
|
trans = m_txLeftStickOrig;
|
|
limitMax = STICK_MAX_MOVE;
|
|
limitMin = STICK_MIN_MOVE;
|
|
move = vertical;
|
|
break;
|
|
case moveRightVerticalStick:
|
|
item = m_txRightStick;
|
|
trans = m_txRightStickOrig;
|
|
limitMax = STICK_MAX_MOVE;
|
|
limitMin = STICK_MIN_MOVE;
|
|
move = vertical;
|
|
break;
|
|
case moveLeftHorizontalStick:
|
|
item = m_txLeftStick;
|
|
trans = m_txLeftStickOrig;
|
|
limitMax = STICK_MAX_MOVE;
|
|
limitMin = STICK_MIN_MOVE;
|
|
move = horizontal;
|
|
break;
|
|
case moveRightHorizontalStick:
|
|
item = m_txRightStick;
|
|
trans = m_txRightStickOrig;
|
|
limitMax = STICK_MAX_MOVE;
|
|
limitMin = STICK_MIN_MOVE;
|
|
move = horizontal;
|
|
break;
|
|
case moveAccess0:
|
|
item = m_txAccess0;
|
|
trans = m_txAccess0Orig;
|
|
limitMax = ACCESS_MAX_MOVE;
|
|
limitMin = ACCESS_MIN_MOVE;
|
|
move = horizontal;
|
|
break;
|
|
case moveAccess1:
|
|
item = m_txAccess1;
|
|
trans = m_txAccess1Orig;
|
|
limitMax = ACCESS_MAX_MOVE;
|
|
limitMin = ACCESS_MIN_MOVE;
|
|
move = horizontal;
|
|
break;
|
|
case moveAccess2:
|
|
item = m_txAccess2;
|
|
trans = m_txAccess2Orig;
|
|
limitMax = ACCESS_MAX_MOVE;
|
|
limitMin = ACCESS_MIN_MOVE;
|
|
move = horizontal;
|
|
break;
|
|
case moveFlightMode:
|
|
item = m_txFlightMode;
|
|
move = jump;
|
|
break;
|
|
case centerAll:
|
|
item = m_txArrows;
|
|
move = jump;
|
|
break;
|
|
case moveAll:
|
|
limitMax = STICK_MAX_MOVE;
|
|
limitMin = STICK_MIN_MOVE;
|
|
move = mix;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (move == vertical) {
|
|
item->setTransform(trans.translate(0, movePos * 10), false);
|
|
} else if (move == horizontal) {
|
|
item->setTransform(trans.translate(movePos * 10, 0), false);
|
|
} else if (move == jump) {
|
|
if (item == m_txArrows) {
|
|
m_txArrows->setVisible(!m_txArrows->isVisible());
|
|
} else if (item == m_txFlightMode) {
|
|
QGraphicsSvgItem *svg;
|
|
svg = (QGraphicsSvgItem *)item;
|
|
if (svg) {
|
|
if (svg->elementId() == "flightModeCenter") {
|
|
if (growing) {
|
|
svg->setElementId("flightModeRight");
|
|
m_txFlightMode->setTransform(m_txFlightModeROrig, false);
|
|
} else {
|
|
svg->setElementId("flightModeLeft");
|
|
m_txFlightMode->setTransform(m_txFlightModeLOrig, false);
|
|
}
|
|
} else if (svg->elementId() == "flightModeRight") {
|
|
growing = false;
|
|
svg->setElementId("flightModeCenter");
|
|
m_txFlightMode->setTransform(m_txFlightModeCOrig, false);
|
|
} else if (svg->elementId() == "flightModeLeft") {
|
|
growing = true;
|
|
svg->setElementId("flightModeCenter");
|
|
m_txFlightMode->setTransform(m_txFlightModeCOrig, false);
|
|
}
|
|
}
|
|
}
|
|
} else if (move == mix) {
|
|
trans = m_txAccess0Orig;
|
|
m_txAccess0->setTransform(trans.translate(movePos * 10 * ACCESS_MAX_MOVE / STICK_MAX_MOVE, 0), false);
|
|
trans = m_txAccess1Orig;
|
|
m_txAccess1->setTransform(trans.translate(movePos * 10 * ACCESS_MAX_MOVE / STICK_MAX_MOVE, 0), false);
|
|
trans = m_txAccess2Orig;
|
|
m_txAccess2->setTransform(trans.translate(movePos * 10 * ACCESS_MAX_MOVE / STICK_MAX_MOVE, 0), false);
|
|
|
|
if (auxFlag) {
|
|
trans = m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(0, movePos * 10), false);
|
|
trans = m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(0, movePos * 10), false);
|
|
} else {
|
|
trans = m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(movePos * 10, 0), false);
|
|
trans = m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(movePos * 10, 0), false);
|
|
}
|
|
|
|
if (movePos == 0) {
|
|
m_txFlightMode->setElementId("flightModeCenter");
|
|
m_txFlightMode->setTransform(m_txFlightModeCOrig, false);
|
|
} else if (movePos == ACCESS_MAX_MOVE / 2) {
|
|
m_txFlightMode->setElementId("flightModeRight");
|
|
m_txFlightMode->setTransform(m_txFlightModeROrig, false);
|
|
} else if (movePos == ACCESS_MIN_MOVE / 2) {
|
|
m_txFlightMode->setElementId("flightModeLeft");
|
|
m_txFlightMode->setTransform(m_txFlightModeLOrig, false);
|
|
}
|
|
}
|
|
if (move == horizontal || move == vertical || move == mix) {
|
|
if (movePos == 0 && growing) {
|
|
auxFlag = !auxFlag;
|
|
}
|
|
if (growing) {
|
|
++movePos;
|
|
} else {
|
|
--movePos;
|
|
}
|
|
if (movePos > limitMax) {
|
|
movePos = movePos - 2;
|
|
growing = false;
|
|
}
|
|
if (movePos < limitMin) {
|
|
movePos = movePos + 2;
|
|
growing = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
AccessoryDesired *ConfigInputWidget::getAccessoryDesiredInstance(int instance)
|
|
{
|
|
switch (instance) {
|
|
case 0:
|
|
if (accessoryDesiredObj0 == NULL) {
|
|
accessoryDesiredObj0 = AccessoryDesired::GetInstance(getObjectManager(), 0);
|
|
}
|
|
return accessoryDesiredObj0;
|
|
|
|
case 1:
|
|
if (accessoryDesiredObj1 == NULL) {
|
|
accessoryDesiredObj1 = AccessoryDesired::GetInstance(getObjectManager(), 1);
|
|
}
|
|
return accessoryDesiredObj1;
|
|
|
|
case 2:
|
|
if (accessoryDesiredObj2 == NULL) {
|
|
accessoryDesiredObj2 = AccessoryDesired::GetInstance(getObjectManager(), 2);
|
|
}
|
|
return accessoryDesiredObj2;
|
|
|
|
default:
|
|
Q_ASSERT(false);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
float ConfigInputWidget::getAccessoryDesiredValue(int instance)
|
|
{
|
|
AccessoryDesired *accessoryDesiredObj = getAccessoryDesiredInstance(instance);
|
|
|
|
if (accessoryDesiredObj == NULL) {
|
|
Q_ASSERT(false);
|
|
return 0.0f;
|
|
}
|
|
|
|
AccessoryDesired::DataFields data = accessoryDesiredObj->getData();
|
|
|
|
return data.AccessoryVal;
|
|
}
|
|
|
|
void ConfigInputWidget::moveSticks()
|
|
{
|
|
QTransform trans;
|
|
|
|
manualCommandData = manualCommandObj->getData();
|
|
flightStatusData = flightStatusObj->getData();
|
|
|
|
switch (transmitterMode) {
|
|
case mode1:
|
|
trans = m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(manualCommandData.Yaw * STICK_MAX_MOVE * 10, manualCommandData.Pitch * STICK_MAX_MOVE * 10), false);
|
|
trans = m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(manualCommandData.Roll * STICK_MAX_MOVE * 10, -manualCommandData.Throttle * STICK_MAX_MOVE * 10), false);
|
|
break;
|
|
case mode2:
|
|
trans = m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(manualCommandData.Yaw * STICK_MAX_MOVE * 10, -manualCommandData.Throttle * STICK_MAX_MOVE * 10), false);
|
|
trans = m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(manualCommandData.Roll * STICK_MAX_MOVE * 10, manualCommandData.Pitch * STICK_MAX_MOVE * 10), false);
|
|
break;
|
|
case mode3:
|
|
trans = m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(manualCommandData.Roll * STICK_MAX_MOVE * 10, manualCommandData.Pitch * STICK_MAX_MOVE * 10), false);
|
|
trans = m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(manualCommandData.Yaw * STICK_MAX_MOVE * 10, -manualCommandData.Throttle * STICK_MAX_MOVE * 10), false);
|
|
break;
|
|
case mode4:
|
|
trans = m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(manualCommandData.Roll * STICK_MAX_MOVE * 10, -manualCommandData.Throttle * STICK_MAX_MOVE * 10), false);
|
|
trans = m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(manualCommandData.Yaw * STICK_MAX_MOVE * 10, manualCommandData.Pitch * STICK_MAX_MOVE * 10), false);
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
break;
|
|
}
|
|
if (flightStatusData.FlightMode == flightModeSettingsData.FlightModePosition[0]) {
|
|
m_txFlightMode->setElementId("flightModeLeft");
|
|
m_txFlightMode->setTransform(m_txFlightModeLOrig, false);
|
|
} else if (flightStatusData.FlightMode == flightModeSettingsData.FlightModePosition[1]) {
|
|
m_txFlightMode->setElementId("flightModeCenter");
|
|
m_txFlightMode->setTransform(m_txFlightModeCOrig, false);
|
|
} else if (flightStatusData.FlightMode == flightModeSettingsData.FlightModePosition[2]) {
|
|
m_txFlightMode->setElementId("flightModeRight");
|
|
m_txFlightMode->setTransform(m_txFlightModeROrig, false);
|
|
}
|
|
|
|
m_txAccess0->setTransform(QTransform(m_txAccess0Orig).translate(getAccessoryDesiredValue(0) * ACCESS_MAX_MOVE * 10, 0), false);
|
|
m_txAccess1->setTransform(QTransform(m_txAccess1Orig).translate(getAccessoryDesiredValue(1) * ACCESS_MAX_MOVE * 10, 0), false);
|
|
m_txAccess2->setTransform(QTransform(m_txAccess2Orig).translate(getAccessoryDesiredValue(2) * ACCESS_MAX_MOVE * 10, 0), false);
|
|
}
|
|
|
|
void ConfigInputWidget::dimOtherControls(bool value)
|
|
{
|
|
qreal opac;
|
|
|
|
if (value) {
|
|
opac = 0.1;
|
|
} else {
|
|
opac = 1;
|
|
}
|
|
m_txAccess0->setOpacity(opac);
|
|
m_txAccess1->setOpacity(opac);
|
|
m_txAccess2->setOpacity(opac);
|
|
m_txFlightMode->setOpacity(opac);
|
|
}
|
|
|
|
void ConfigInputWidget::invertControls()
|
|
{
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
foreach(QWidget * wd, extraWidgets) {
|
|
QCheckBox *cb = qobject_cast<QCheckBox *>(wd);
|
|
|
|
if (cb) {
|
|
int index = manualSettingsObj->getField("ChannelNumber")->getElementNames().indexOf(cb->text());
|
|
if ((cb->isChecked() && (manualSettingsData.ChannelMax[index] > manualSettingsData.ChannelMin[index])) ||
|
|
(!cb->isChecked() && (manualSettingsData.ChannelMax[index] < manualSettingsData.ChannelMin[index]))) {
|
|
qint16 aux;
|
|
aux = manualSettingsData.ChannelMax[index];
|
|
manualSettingsData.ChannelMax[index] = manualSettingsData.ChannelMin[index];
|
|
manualSettingsData.ChannelMin[index] = aux;
|
|
}
|
|
}
|
|
}
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
}
|
|
|
|
void ConfigInputWidget::moveFMSlider()
|
|
{
|
|
ManualControlSettings::DataFields manualSettingsDataPriv = manualSettingsObj->getData();
|
|
ManualControlCommand::DataFields manualCommandDataPriv = manualCommandObj->getData();
|
|
|
|
float valueScaled;
|
|
int chMin = manualSettingsDataPriv.ChannelMin[ManualControlSettings::CHANNELMIN_FLIGHTMODE];
|
|
int chMax = manualSettingsDataPriv.ChannelMax[ManualControlSettings::CHANNELMAX_FLIGHTMODE];
|
|
int chNeutral = manualSettingsDataPriv.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_FLIGHTMODE];
|
|
|
|
int value = manualCommandDataPriv.Channel[ManualControlSettings::CHANNELMIN_FLIGHTMODE];
|
|
|
|
if ((chMax > chMin && value >= chNeutral) || (chMin > chMax && value <= chNeutral)) {
|
|
if (chMax != chNeutral) {
|
|
valueScaled = (float)(value - chNeutral) / (float)(chMax - chNeutral);
|
|
} else {
|
|
valueScaled = 0;
|
|
}
|
|
} else {
|
|
if (chMin != chNeutral) {
|
|
valueScaled = (float)(value - chNeutral) / (float)(chNeutral - chMin);
|
|
} else {
|
|
valueScaled = 0;
|
|
}
|
|
}
|
|
|
|
// Bound and scale FlightMode from [-1..+1] to [0..1] range
|
|
if (valueScaled < -1.0) {
|
|
valueScaled = -1.0;
|
|
} else if (valueScaled > 1.0) {
|
|
valueScaled = 1.0;
|
|
}
|
|
|
|
// Convert flightMode value into the switch position in the range [0..N-1]
|
|
// This uses the same optimized computation as flight code to be consistent
|
|
uint8_t pos = ((int16_t)(valueScaled * 256) + 256) * manualSettingsDataPriv.FlightModeNumber >> 9;
|
|
if (pos >= manualSettingsDataPriv.FlightModeNumber) {
|
|
pos = manualSettingsDataPriv.FlightModeNumber - 1;
|
|
}
|
|
ui->fmsSlider->setValue(pos);
|
|
}
|
|
|
|
void ConfigInputWidget::updatePositionSlider()
|
|
{
|
|
ManualControlSettings::DataFields manualSettingsDataPriv = manualSettingsObj->getData();
|
|
|
|
switch (manualSettingsDataPriv.FlightModeNumber) {
|
|
default:
|
|
case 6:
|
|
ui->fmsModePos6->setEnabled(true);
|
|
ui->pidBankSs1_5->setEnabled(true);
|
|
ui->assistControlPos6->setEnabled(true);
|
|
// pass through
|
|
case 5:
|
|
ui->fmsModePos5->setEnabled(true);
|
|
ui->pidBankSs1_4->setEnabled(true);
|
|
ui->assistControlPos5->setEnabled(true);
|
|
// pass through
|
|
case 4:
|
|
ui->fmsModePos4->setEnabled(true);
|
|
ui->pidBankSs1_3->setEnabled(true);
|
|
ui->assistControlPos4->setEnabled(true);
|
|
// pass through
|
|
case 3:
|
|
ui->fmsModePos3->setEnabled(true);
|
|
ui->pidBankSs1_2->setEnabled(true);
|
|
ui->assistControlPos3->setEnabled(true);
|
|
// pass through
|
|
case 2:
|
|
ui->fmsModePos2->setEnabled(true);
|
|
ui->pidBankSs1_1->setEnabled(true);
|
|
ui->assistControlPos2->setEnabled(true);
|
|
// pass through
|
|
case 1:
|
|
ui->fmsModePos1->setEnabled(true);
|
|
ui->pidBankSs1_0->setEnabled(true);
|
|
ui->assistControlPos1->setEnabled(true);
|
|
// pass through
|
|
case 0:
|
|
break;
|
|
}
|
|
|
|
switch (manualSettingsDataPriv.FlightModeNumber) {
|
|
case 0:
|
|
ui->fmsModePos1->setEnabled(false);
|
|
ui->pidBankSs1_0->setEnabled(false);
|
|
ui->assistControlPos1->setEnabled(false);
|
|
// pass through
|
|
case 1:
|
|
ui->fmsModePos2->setEnabled(false);
|
|
ui->pidBankSs1_1->setEnabled(false);
|
|
ui->assistControlPos2->setEnabled(false);
|
|
// pass through
|
|
case 2:
|
|
ui->fmsModePos3->setEnabled(false);
|
|
ui->pidBankSs1_2->setEnabled(false);
|
|
ui->assistControlPos3->setEnabled(false);
|
|
// pass through
|
|
case 3:
|
|
ui->fmsModePos4->setEnabled(false);
|
|
ui->pidBankSs1_3->setEnabled(false);
|
|
ui->assistControlPos4->setEnabled(false);
|
|
// pass through
|
|
case 4:
|
|
ui->fmsModePos5->setEnabled(false);
|
|
ui->pidBankSs1_4->setEnabled(false);
|
|
ui->assistControlPos5->setEnabled(false);
|
|
// pass through
|
|
case 5:
|
|
ui->fmsModePos6->setEnabled(false);
|
|
ui->pidBankSs1_5->setEnabled(false);
|
|
ui->assistControlPos6->setEnabled(false);
|
|
// pass through
|
|
case 6:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
QString fmNumber = QString().setNum(manualSettingsDataPriv.FlightModeNumber);
|
|
int count = 0;
|
|
foreach(QSlider * sp, findChildren<QSlider *>()) {
|
|
// Find FlightMode slider and apply stylesheet
|
|
if (sp->objectName() == "channelNeutral") {
|
|
if (count == 4) {
|
|
sp->setStyleSheet(
|
|
"QSlider::groove:horizontal {border: 2px solid rgb(196, 196, 196); height: 12px; border-radius: 4px; "
|
|
"border-image:url(:/configgadget/images/flightmode_bg" + fmNumber + ".png); }"
|
|
"QSlider::add-page:horizontal { background: none; border: none; }"
|
|
"QSlider::sub-page:horizontal { background: none; border: none; }"
|
|
"QSlider::handle:horizontal { background: rgba(196, 196, 196, 255); width: 10px; height: 28px; "
|
|
"margin: -3px -2px; border-radius: 3px; border: 1px solid #777; }");
|
|
count++;
|
|
} else {
|
|
count++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::updateCalibration()
|
|
{
|
|
manualCommandData = manualCommandObj->getData();
|
|
for (uint i = 0; i < ManualControlSettings::CHANNELMAX_NUMELEM; ++i) {
|
|
if ((!reverse[i] && manualSettingsData.ChannelMin[i] > manualCommandData.Channel[i]) ||
|
|
(reverse[i] && manualSettingsData.ChannelMin[i] < manualCommandData.Channel[i])) {
|
|
manualSettingsData.ChannelMin[i] = manualCommandData.Channel[i];
|
|
}
|
|
if ((!reverse[i] && manualSettingsData.ChannelMax[i] < manualCommandData.Channel[i]) ||
|
|
(reverse[i] && manualSettingsData.ChannelMax[i] > manualCommandData.Channel[i])) {
|
|
manualSettingsData.ChannelMax[i] = manualCommandData.Channel[i];
|
|
}
|
|
if (i == ManualControlSettings::CHANNELNUMBER_FLIGHTMODE || i == ManualControlSettings::CHANNELNUMBER_FLIGHTMODE) {
|
|
adjustSpecialNeutrals();
|
|
} else {
|
|
manualSettingsData.ChannelNeutral[i] = manualCommandData.Channel[i];
|
|
}
|
|
}
|
|
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
manualSettingsObj->updated();
|
|
}
|
|
|
|
void ConfigInputWidget::simpleCalibration(bool enable)
|
|
{
|
|
if (enable) {
|
|
ui->configurationWizard->setEnabled(false);
|
|
ui->saveRCInputToRAM->setEnabled(false);
|
|
ui->saveRCInputToSD->setEnabled(false);
|
|
ui->runCalibration->setText(tr("Stop Manual Calibration"));
|
|
|
|
QMessageBox msgBox;
|
|
msgBox.setText(tr("<p>Arming Settings are now set to 'Always Disarmed' for your safety.</p>"
|
|
"<p>Be sure your receiver is powered with an external source and Transmitter is on.</p>"
|
|
"<p align='center'><b>Stop Manual Calibration</b> when done</p>"));
|
|
msgBox.setDetailedText(tr("You will have to reconfigure the arming settings manually when the manual calibration is finished."));
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.setDefaultButton(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
|
|
manualCommandData = manualCommandObj->getData();
|
|
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
flightModeSettingsData = flightModeSettingsObj->getData();
|
|
flightModeSettingsData.Arming = FlightModeSettings::ARMING_ALWAYSDISARMED;
|
|
flightModeSettingsObj->setData(flightModeSettingsData);
|
|
|
|
for (unsigned int i = 0; i < ManualControlCommand::CHANNEL_NUMELEM; i++) {
|
|
reverse[i] = manualSettingsData.ChannelMax[i] < manualSettingsData.ChannelMin[i];
|
|
manualSettingsData.ChannelMin[i] = manualCommandData.Channel[i];
|
|
manualSettingsData.ChannelNeutral[i] = manualCommandData.Channel[i];
|
|
manualSettingsData.ChannelMax[i] = manualCommandData.Channel[i];
|
|
}
|
|
|
|
fastMdataSingle(manualCommandObj, &manualControlMdata);
|
|
|
|
// Stash actuatorSettings
|
|
actuatorSettingsData = actuatorSettingsObj->getData();
|
|
previousActuatorSettingsData = actuatorSettingsData;
|
|
|
|
// Disable all actuators
|
|
resetActuatorSettings();
|
|
|
|
connect(manualCommandObj, SIGNAL(objectUnpacked(UAVObject *)), this, SLOT(updateCalibration()));
|
|
} else {
|
|
ui->configurationWizard->setEnabled(true);
|
|
ui->saveRCInputToRAM->setEnabled(true);
|
|
ui->saveRCInputToSD->setEnabled(true);
|
|
ui->runCalibration->setText(tr("Start Manual Calibration"));
|
|
|
|
manualCommandData = manualCommandObj->getData();
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
|
|
restoreMdataSingle(manualCommandObj, &manualControlMdata);
|
|
|
|
for (unsigned int i = 0; i < ManualControlCommand::CHANNEL_NUMELEM; i++) {
|
|
if (i == ManualControlSettings::CHANNELNUMBER_FLIGHTMODE || i == ManualControlSettings::CHANNELNUMBER_FLIGHTMODE) {
|
|
adjustSpecialNeutrals();
|
|
} else {
|
|
manualSettingsData.ChannelNeutral[i] = manualCommandData.Channel[i];
|
|
}
|
|
}
|
|
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
|
|
// Load actuator settings back from beginning of manual calibration
|
|
actuatorSettingsObj->setData(previousActuatorSettingsData);
|
|
|
|
disconnect(manualCommandObj, SIGNAL(objectUnpacked(UAVObject *)), this, SLOT(updateCalibration()));
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::adjustSpecialNeutrals()
|
|
{
|
|
// FlightMode and Throttle need special neutral settings
|
|
//
|
|
// Force flight mode neutral to middle
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNUMBER_FLIGHTMODE] =
|
|
(manualSettingsData.ChannelMax[ManualControlSettings::CHANNELNUMBER_FLIGHTMODE] +
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELNUMBER_FLIGHTMODE]) / 2;
|
|
|
|
// Force throttle to be near min, add 4% from total range to avoid arming issues
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_THROTTLE] =
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE] +
|
|
((manualSettingsData.ChannelMax[ManualControlSettings::CHANNELMAX_THROTTLE] -
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE]) * 0.04);
|
|
}
|
|
|
|
bool ConfigInputWidget::shouldObjectBeSaved(UAVObject *object)
|
|
{
|
|
// ManualControlCommand no need to be saved
|
|
return dynamic_cast<ManualControlCommand *>(object) == NULL;
|
|
}
|
|
|
|
void ConfigInputWidget::resetChannelSettings()
|
|
{
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
// Clear all channel data : Channel Type (PPM,PWM..) and Number
|
|
for (unsigned int channel = 0; channel < 9; channel++) {
|
|
manualSettingsData.ChannelGroups[channel] = ManualControlSettings::CHANNELGROUPS_NONE;
|
|
manualSettingsData.ChannelNumber[channel] = 0;
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::resetActuatorSettings()
|
|
{
|
|
actuatorSettingsData = actuatorSettingsObj->getData();
|
|
// Clear all output data : Min, max, neutral = 1500
|
|
// 1500 = servo middle, can be applied to all outputs because board is 'Alwaysdisarmed'
|
|
for (unsigned int output = 0; output < 12; output++) {
|
|
actuatorSettingsData.ChannelMax[output] = 1500;
|
|
actuatorSettingsData.ChannelMin[output] = 1000;
|
|
actuatorSettingsData.ChannelNeutral[output] = 1500;
|
|
actuatorSettingsObj->setData(actuatorSettingsData);
|
|
}
|
|
}
|