mirror of
https://bitbucket.org/librepilot/librepilot.git
synced 2025-01-24 09:52:11 +01:00
1291 lines
49 KiB
C++
1291 lines
49 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 "uavtalk/telemetrymanager.h"
|
|
|
|
#include <QDebug>
|
|
#include <QStringList>
|
|
#include <QtGui/QWidget>
|
|
#include <QtGui/QTextEdit>
|
|
#include <QtGui/QVBoxLayout>
|
|
#include <QtGui/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),transmitterType(heli),loop(NULL),skipflag(false)
|
|
{
|
|
manualCommandObj = ManualControlCommand::GetInstance(getObjectManager());
|
|
manualSettingsObj = ManualControlSettings::GetInstance(getObjectManager());
|
|
receiverActivityObj=ReceiverActivity::GetInstance(getObjectManager());
|
|
m_config = new Ui_InputWidget();
|
|
m_config->setupUi(this);
|
|
|
|
addApplySaveButtons(m_config->saveRCInputToRAM,m_config->saveRCInputToSD);
|
|
|
|
//Generate the rows of buttons in the input channel form GUI
|
|
unsigned int index=0;
|
|
foreach (QString name, manualSettingsObj->getField("ChannelNumber")->getElementNames())
|
|
{
|
|
Q_ASSERT(index < ManualControlSettings::CHANNELGROUPS_NUMELEM);
|
|
inputChannelForm * inpForm=new inputChannelForm(this,index==0);
|
|
m_config->channelSettings->layout()->addWidget(inpForm); //Add the row to the UI
|
|
inpForm->setName(name);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","ChannelGroups",inpForm->ui->channelGroup,index);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","ChannelNumber",inpForm->ui->channelNumber,index);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","ChannelMin",inpForm->ui->channelMin,index);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","ChannelNeutral",inpForm->ui->channelNeutral,index);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","ChannelMax",inpForm->ui->channelMax,index);
|
|
++index;
|
|
}
|
|
|
|
addUAVObjectToWidgetRelation("ManualControlSettings", "Deadband", m_config->deadband, 0, 0.01f);
|
|
|
|
connect(m_config->configurationWizard,SIGNAL(clicked()),this,SLOT(goToWizard()));
|
|
connect(m_config->runCalibration,SIGNAL(toggled(bool)),this, SLOT(simpleCalibration(bool)));
|
|
|
|
connect(m_config->wzNext,SIGNAL(clicked()),this,SLOT(wzNext()));
|
|
connect(m_config->wzCancel,SIGNAL(clicked()),this,SLOT(wzCancel()));
|
|
connect(m_config->wzBack,SIGNAL(clicked()),this,SLOT(wzBack()));
|
|
|
|
m_config->stackedWidget->setCurrentIndex(0);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModePosition",m_config->fmsModePos1,0,1,true);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModePosition",m_config->fmsModePos2,1,1,true);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModePosition",m_config->fmsModePos3,2,1,true);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModePosition",m_config->fmsModePos4,3,1,true);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModePosition",m_config->fmsModePos5,4,1,true);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModePosition",m_config->fmsModePos6,5,1,true);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","FlightModeNumber",m_config->fmsPosNum);
|
|
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization1Settings",m_config->fmsSsPos1Roll,"Roll");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization2Settings",m_config->fmsSsPos2Roll,"Roll");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization3Settings",m_config->fmsSsPos3Roll,"Roll");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization1Settings",m_config->fmsSsPos1Pitch,"Pitch");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization2Settings",m_config->fmsSsPos2Pitch,"Pitch");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization3Settings",m_config->fmsSsPos3Pitch,"Pitch");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization1Settings",m_config->fmsSsPos1Yaw,"Yaw");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization2Settings",m_config->fmsSsPos2Yaw,"Yaw");
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Stabilization3Settings",m_config->fmsSsPos3Yaw,"Yaw");
|
|
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","Arming",m_config->armControl);
|
|
addUAVObjectToWidgetRelation("ManualControlSettings","ArmedTimeout",m_config->armTimeout,0,1000);
|
|
connect( ManualControlCommand::GetInstance(getObjectManager()),SIGNAL(objectUpdated(UAVObject*)),this,SLOT(moveFMSlider()));
|
|
connect( ManualControlSettings::GetInstance(getObjectManager()),SIGNAL(objectUpdated(UAVObject*)),this,SLOT(updatePositionSlider()));
|
|
enableControls(false);
|
|
|
|
populateWidgets();
|
|
refreshWidgetsValues();
|
|
// Connect the help button
|
|
connect(m_config->inputHelp, SIGNAL(clicked()), this, SLOT(openHelp()));
|
|
|
|
m_config->graphicsView->setScene(new QGraphicsScene(this));
|
|
m_config->graphicsView->setViewportUpdateMode(QGraphicsView::FullViewportUpdate);
|
|
m_renderer = new QSvgRenderer();
|
|
QGraphicsScene *l_scene = m_config->graphicsView->scene();
|
|
m_config->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);
|
|
}
|
|
m_config->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;
|
|
}
|
|
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::resizeEvent(QResizeEvent *event)
|
|
{
|
|
QWidget::resizeEvent(event);
|
|
m_config->graphicsView->fitInView(m_txBackground, Qt::KeepAspectRatio );
|
|
}
|
|
|
|
void ConfigInputWidget::openHelp()
|
|
{
|
|
|
|
QDesktopServices::openUrl( QUrl("http://wiki.openpilot.org/display/Doc/Input+Configuration", 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 arming settings yourself afterwards."));
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.setDefaultButton(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
wizardSetUpStep(wizardWelcome);
|
|
m_config->graphicsView->fitInView(m_txBackground, Qt::KeepAspectRatio );
|
|
}
|
|
|
|
void ConfigInputWidget::wzCancel()
|
|
{
|
|
dimOtherControls(false);
|
|
manualCommandObj->setMetadata(manualCommandObj->getDefaultMetadata());
|
|
m_config->stackedWidget->setCurrentIndex(0);
|
|
|
|
if(wizardStep != wizardNone)
|
|
wizardTearDownStep(wizardStep);
|
|
wizardStep=wizardNone;
|
|
m_config->stackedWidget->setCurrentIndex(0);
|
|
|
|
// Load settings back from beginning of wizard
|
|
manualSettingsObj->setData(previousManualSettingsData);
|
|
}
|
|
|
|
void ConfigInputWidget::wzNext()
|
|
{
|
|
// 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(wizardChooseMode);
|
|
break;
|
|
case wizardChooseMode:
|
|
wizardSetUpStep(wizardChooseType);
|
|
break;
|
|
case wizardChooseType:
|
|
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;
|
|
m_config->stackedWidget->setCurrentIndex(0);
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::wzBack()
|
|
{
|
|
if(wizardStep != wizardNone &&
|
|
wizardStep != wizardIdentifySticks)
|
|
wizardTearDownStep(wizardStep);
|
|
|
|
// State transitions for next button
|
|
switch(wizardStep) {
|
|
case wizardChooseMode:
|
|
wizardSetUpStep(wizardWelcome);
|
|
break;
|
|
case wizardChooseType:
|
|
wizardSetUpStep(wizardChooseMode);
|
|
break;
|
|
case wizardIdentifySticks:
|
|
prevChannel();
|
|
if(currentChannelNum == -1) {
|
|
wizardTearDownStep(wizardIdentifySticks);
|
|
wizardSetUpStep(wizardChooseType);
|
|
}
|
|
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)
|
|
{
|
|
switch(step) {
|
|
case wizardWelcome:
|
|
m_config->graphicsView->setVisible(false);
|
|
setTxMovement(nothing);
|
|
manualSettingsData=manualSettingsObj->getData();
|
|
manualSettingsData.Arming=ManualControlSettings::ARMING_ALWAYSDISARMED;
|
|
previousManualSettingsData = manualSettingsData;
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
m_config->wzText->setText(tr("Welcome to the inputs configuration wizard.\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"
|
|
"At any time you can press 'back' to return to the previous screeen or 'Cancel' to cancel the wizard.\n"));
|
|
m_config->stackedWidget->setCurrentIndex(1);
|
|
m_config->wzBack->setEnabled(false);
|
|
break;
|
|
case wizardChooseMode:
|
|
{
|
|
m_config->graphicsView->setVisible(true);
|
|
m_config->graphicsView->fitInView(m_txBackground, Qt::KeepAspectRatio );
|
|
setTxMovement(nothing);
|
|
m_config->wzText->setText(tr("Please choose your transmiter type.\n"
|
|
"Mode 1 means your throttle stick is on the right\n"
|
|
"Mode 2 means your throttle stick is on the left\n"));
|
|
m_config->wzBack->setEnabled(true);
|
|
QRadioButton * mode1=new QRadioButton(tr("Mode 1"),this);
|
|
QRadioButton * mode2=new QRadioButton(tr("Mode 2"),this);
|
|
mode2->setChecked(true);
|
|
extraWidgets.clear();
|
|
extraWidgets.append(mode1);
|
|
extraWidgets.append(mode2);
|
|
m_config->checkBoxesLayout->layout()->addWidget(mode1);
|
|
m_config->checkBoxesLayout->layout()->addWidget(mode2);
|
|
}
|
|
break;
|
|
case wizardChooseType:
|
|
{
|
|
m_config->wzText->setText(tr("Please choose your transmiter mode.\n"
|
|
"Acro means normal transmitter\n"
|
|
"Heli means there is a collective pitch and throttle input\n"
|
|
"If you are using a heli transmitter please engage throttle hold now please.\n"));
|
|
m_config->wzBack->setEnabled(true);
|
|
QRadioButton * typeAcro=new QRadioButton(tr("Acro"),this);
|
|
QRadioButton * typeHeli=new QRadioButton(tr("Heli"),this);
|
|
typeAcro->setChecked(true);
|
|
typeHeli->setChecked(false);
|
|
extraWidgets.clear();
|
|
extraWidgets.append(typeAcro);
|
|
extraWidgets.append(typeHeli);
|
|
m_config->checkBoxesLayout->layout()->addWidget(typeAcro);
|
|
m_config->checkBoxesLayout->layout()->addWidget(typeHeli);
|
|
wizardStep=wizardChooseType;
|
|
}
|
|
break;
|
|
case wizardIdentifySticks:
|
|
usedChannels.clear();
|
|
currentChannelNum=-1;
|
|
nextChannel();
|
|
manualSettingsData=manualSettingsObj->getData();
|
|
connect(receiverActivityObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(identifyControls()));
|
|
m_config->wzNext->setEnabled(false);
|
|
break;
|
|
case wizardIdentifyCenter:
|
|
setTxMovement(centerAll);
|
|
m_config->wzText->setText(QString(tr("Please center all control controls and press next when ready (if your FlightMode switch has only two positions, leave it on either position)")));
|
|
break;
|
|
case wizardIdentifyLimits:
|
|
{
|
|
setTxMovement(nothing);
|
|
m_config->wzText->setText(QString(tr("Please move all controls to their maximum extents on both directions and press next when ready")));
|
|
fastMdata();
|
|
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()));
|
|
}
|
|
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"))
|
|
{
|
|
QCheckBox * cb=new QCheckBox(name,this);
|
|
// Make sure checked status matches current one
|
|
cb->setChecked(manualSettingsData.ChannelMax[index] < manualSettingsData.ChannelMin[index]);
|
|
|
|
extraWidgets.append(cb);
|
|
m_config->checkBoxesLayout->layout()->addWidget(cb);
|
|
|
|
connect(cb,SIGNAL(toggled(bool)),this,SLOT(invertControls()));
|
|
}
|
|
}
|
|
connect(manualCommandObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(moveSticks()));
|
|
m_config->wzText->setText(QString(tr("Please check the picture below and check all the sticks which show an inverted movement and press next when ready")));
|
|
fastMdata();
|
|
break;
|
|
case wizardFinish:
|
|
dimOtherControls(true);
|
|
connect(manualCommandObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(moveSticks()));
|
|
m_config->wzText->setText(QString(tr("You have completed this wizard, please check below if the picture below mimics your sticks movement.\n"
|
|
"This new settings aren't saved to the board yet, after pressing next you will go to the initial screen where you can do that.")));
|
|
fastMdata();
|
|
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_THROTTLE]=
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE]+
|
|
((manualSettingsData.ChannelMax[ManualControlSettings::CHANNELMAX_THROTTLE]-
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE])*0.02);
|
|
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);
|
|
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 wizardChooseMode:
|
|
mode=qobject_cast<QRadioButton *>(extraWidgets.at(0));
|
|
if(mode->isChecked())
|
|
transmitterMode=mode1;
|
|
else
|
|
transmitterMode=mode2;
|
|
delete extraWidgets.at(0);
|
|
delete extraWidgets.at(1);
|
|
extraWidgets.clear();
|
|
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 wizardIdentifySticks:
|
|
disconnect(receiverActivityObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(identifyControls()));
|
|
m_config->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()));
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
restoreMdata();
|
|
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()));
|
|
restoreMdata();
|
|
break;
|
|
case wizardFinish:
|
|
dimOtherControls(false);
|
|
setTxMovement(nothing);
|
|
disconnect(manualCommandObj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(moveSticks()));
|
|
restoreMdata();
|
|
break;
|
|
default:
|
|
Q_ASSERT(0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set manual control command to fast updates
|
|
*/
|
|
void ConfigInputWidget::fastMdata()
|
|
{
|
|
manualControlMdata = manualCommandObj->getMetadata();
|
|
UAVObject::Metadata mdata = manualControlMdata;
|
|
UAVObject::SetFlightTelemetryUpdateMode(mdata, UAVObject::UPDATEMODE_PERIODIC);
|
|
mdata.flightTelemetryUpdatePeriod = 150;
|
|
manualCommandObj->setMetadata(mdata);
|
|
}
|
|
|
|
/**
|
|
* Restore previous update settings for manual control data
|
|
*/
|
|
void ConfigInputWidget::restoreMdata()
|
|
{
|
|
manualCommandObj->setMetadata(manualControlMdata);
|
|
}
|
|
|
|
/**
|
|
* Set the display to indicate which channel the person should move
|
|
*/
|
|
void ConfigInputWidget::setChannel(int newChan)
|
|
{
|
|
if(newChan == ManualControlSettings::CHANNELGROUPS_COLLECTIVE)
|
|
m_config->wzText->setText(QString(tr("Please enable throttle hold mode and move the collective pitch stick")));
|
|
else if (newChan == ManualControlSettings::CHANNELGROUPS_FLIGHTMODE)
|
|
m_config->wzText->setText(QString(tr("Please flick the flight mode switch. For switches you may have to repeat this rapidly.")));
|
|
else if((transmitterType == heli) && (newChan == ManualControlSettings::CHANNELGROUPS_THROTTLE))
|
|
m_config->wzText->setText(QString(tr("Please disable throttle hold mode and move the throttle stick")));
|
|
else
|
|
m_config->wzText->setText(QString(tr("Please move each control once 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") ||
|
|
manualSettingsObj->getField("ChannelGroups")->getElementNames().at(newChan).contains("FlightMode")) {
|
|
m_config->wzNext->setEnabled(true);
|
|
m_config->wzText->setText(m_config->wzText->text() + tr(" or click next to skip this channel."));
|
|
} else
|
|
m_config->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) {
|
|
setChannel(order[i-1]);
|
|
usedChannels.removeLast();
|
|
return;
|
|
}
|
|
}
|
|
currentChannelNum = -1; // hit end of list
|
|
}
|
|
|
|
void ConfigInputWidget::identifyControls()
|
|
{
|
|
static int debounce=0;
|
|
|
|
receiverActivityData=receiverActivityObj->getData();
|
|
if(receiverActivityData.ActiveChannel==255)
|
|
return;
|
|
if(channelDetected)
|
|
return;
|
|
else
|
|
{
|
|
receiverActivityData=receiverActivityObj->getData();
|
|
currentChannel.group=receiverActivityData.ActiveGroup;
|
|
currentChannel.number=receiverActivityData.ActiveChannel;
|
|
if(currentChannel==lastChannel)
|
|
++debounce;
|
|
lastChannel.group= currentChannel.group;
|
|
lastChannel.number=currentChannel.number;
|
|
if(!usedChannels.contains(lastChannel) && debounce>1)
|
|
{
|
|
channelDetected = true;
|
|
debounce=0;
|
|
usedChannels.append(lastChannel);
|
|
manualSettingsData=manualSettingsObj->getData();
|
|
manualSettingsData.ChannelGroups[currentChannelNum]=currentChannel.group;
|
|
manualSettingsData.ChannelNumber[currentChannelNum]=currentChannel.number;
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
}
|
|
else
|
|
return;
|
|
}
|
|
|
|
m_config->wzText->clear();
|
|
setTxMovement(nothing);
|
|
|
|
QTimer::singleShot(2500, this, SLOT(wzNext()));
|
|
}
|
|
|
|
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)
|
|
{
|
|
//CHANNELNUMBER_ROLL=0, CHANNELNUMBER_PITCH=1, CHANNELNUMBER_YAW=2, CHANNELNUMBER_THROTTLE=3, CHANNELNUMBER_FLIGHTMODE=4, CHANNELNUMBER_ACCESSORY0=5, CHANNELNUMBER_ACCESSORY1=6, CHANNELNUMBER_ACCESSORY2=7 } ChannelNumberElem;
|
|
if(command==ManualControlSettings::CHANNELNUMBER_ROLL)
|
|
{
|
|
setTxMovement(moveRightHorizontalStick);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_PITCH)
|
|
{
|
|
if(transmitterMode==mode2)
|
|
setTxMovement(moveRightVerticalStick);
|
|
else
|
|
setTxMovement(moveLeftVerticalStick);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_YAW)
|
|
{
|
|
setTxMovement(moveLeftHorizontalStick);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_THROTTLE)
|
|
{
|
|
if(transmitterMode==mode2)
|
|
setTxMovement(moveLeftVerticalStick);
|
|
else
|
|
setTxMovement(moveRightVerticalStick);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_COLLECTIVE)
|
|
{
|
|
if(transmitterMode==mode2)
|
|
setTxMovement(moveLeftVerticalStick);
|
|
else
|
|
setTxMovement(moveRightVerticalStick);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_FLIGHTMODE)
|
|
{
|
|
setTxMovement(moveFlightMode);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_ACCESSORY0)
|
|
{
|
|
setTxMovement(moveAccess0);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_ACCESSORY1)
|
|
{
|
|
setTxMovement(moveAccess1);
|
|
}
|
|
else if(command==ManualControlSettings::CHANNELNUMBER_ACCESSORY2)
|
|
{
|
|
setTxMovement(moveAccess2);
|
|
}
|
|
|
|
}
|
|
|
|
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:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::moveTxControls()
|
|
{
|
|
QTransform trans;
|
|
QGraphicsItem * item;
|
|
txMovementType move;
|
|
int limitMax;
|
|
int limitMin;
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConfigInputWidget::moveSticks()
|
|
{
|
|
QTransform trans;
|
|
manualCommandData=manualCommandObj->getData();
|
|
if(transmitterMode==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);
|
|
}
|
|
else
|
|
{
|
|
trans=m_txRightStickOrig;
|
|
m_txRightStick->setTransform(trans.translate(manualCommandData.Roll*STICK_MAX_MOVE*10,-manualCommandData.Throttle*STICK_MAX_MOVE*10),false);
|
|
trans=m_txLeftStickOrig;
|
|
m_txLeftStick->setTransform(trans.translate(manualCommandData.Yaw*STICK_MAX_MOVE*10,manualCommandData.Pitch*STICK_MAX_MOVE*10),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::enableControls(bool enable)
|
|
{
|
|
m_config->configurationWizard->setEnabled(enable);
|
|
m_config->runCalibration->setEnabled(enable);
|
|
|
|
ConfigTaskWidget::enableControls(enable);
|
|
|
|
}
|
|
|
|
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;
|
|
m_config->fmsSlider->setValue(pos);
|
|
}
|
|
|
|
void ConfigInputWidget::updatePositionSlider()
|
|
{
|
|
ManualControlSettings::DataFields manualSettingsDataPriv = manualSettingsObj->getData();
|
|
|
|
switch(manualSettingsDataPriv.FlightModeNumber) {
|
|
default:
|
|
case 6:
|
|
m_config->fmsModePos6->setEnabled(true);
|
|
// pass through
|
|
case 5:
|
|
m_config->fmsModePos5->setEnabled(true);
|
|
// pass through
|
|
case 4:
|
|
m_config->fmsModePos4->setEnabled(true);
|
|
// pass through
|
|
case 3:
|
|
m_config->fmsModePos3->setEnabled(true);
|
|
// pass through
|
|
case 2:
|
|
m_config->fmsModePos2->setEnabled(true);
|
|
// pass through
|
|
case 1:
|
|
m_config->fmsModePos1->setEnabled(true);
|
|
// pass through
|
|
case 0:
|
|
break;
|
|
}
|
|
|
|
switch(manualSettingsDataPriv.FlightModeNumber) {
|
|
case 0:
|
|
m_config->fmsModePos1->setEnabled(false);
|
|
// pass through
|
|
case 1:
|
|
m_config->fmsModePos2->setEnabled(false);
|
|
// pass through
|
|
case 2:
|
|
m_config->fmsModePos3->setEnabled(false);
|
|
// pass through
|
|
case 3:
|
|
m_config->fmsModePos4->setEnabled(false);
|
|
// pass through
|
|
case 4:
|
|
m_config->fmsModePos5->setEnabled(false);
|
|
// pass through
|
|
case 5:
|
|
m_config->fmsModePos6->setEnabled(false);
|
|
// pass through
|
|
case 6:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
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];
|
|
manualSettingsData.ChannelNeutral[i] = manualCommandData.Channel[i];
|
|
}
|
|
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
manualSettingsObj->updated();
|
|
}
|
|
|
|
void ConfigInputWidget::simpleCalibration(bool enable)
|
|
{
|
|
if (enable) {
|
|
m_config->configurationWizard->setEnabled(false);
|
|
|
|
QMessageBox msgBox;
|
|
msgBox.setText(tr("Arming Settings are now set to Always Disarmed for your safety."));
|
|
msgBox.setDetailedText(tr("You will have to reconfigure arming settings yourself afterwards."));
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
msgBox.setDefaultButton(QMessageBox::Ok);
|
|
msgBox.exec();
|
|
|
|
manualCommandData = manualCommandObj->getData();
|
|
|
|
manualSettingsData=manualSettingsObj->getData();
|
|
manualSettingsData.Arming=ManualControlSettings::ARMING_ALWAYSDISARMED;
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
|
|
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];
|
|
}
|
|
|
|
fastMdata();
|
|
|
|
connect(manualCommandObj, SIGNAL(objectUnpacked(UAVObject*)), this, SLOT(updateCalibration()));
|
|
} else {
|
|
m_config->configurationWizard->setEnabled(true);
|
|
|
|
manualCommandData = manualCommandObj->getData();
|
|
manualSettingsData = manualSettingsObj->getData();
|
|
|
|
restoreMdata();
|
|
|
|
for (int i = 0; i < ManualControlCommand::CHANNEL_NUMELEM; i++)
|
|
manualSettingsData.ChannelNeutral[i] = manualCommandData.Channel[i];
|
|
|
|
// 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
|
|
manualSettingsData.ChannelNeutral[ManualControlSettings::CHANNELNEUTRAL_THROTTLE]=
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE]+
|
|
((manualSettingsData.ChannelMax[ManualControlSettings::CHANNELMAX_THROTTLE]-
|
|
manualSettingsData.ChannelMin[ManualControlSettings::CHANNELMIN_THROTTLE])*0.02);
|
|
|
|
manualSettingsObj->setData(manualSettingsData);
|
|
|
|
disconnect(manualCommandObj, SIGNAL(objectUnpacked(UAVObject*)), this, SLOT(updateCalibration()));
|
|
}
|
|
}
|