/** ****************************************************************************** * * @file configservowidget.cpp * @author E. Lafargue & 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 "configoutputwidget.h" #include "uavtalk/telemetrymanager.h" #include #include #include #include #include #include #include #include #include #include "actuatorcommand.h" #include "systemalarms.h" #include "uavsettingsimportexport/uavsettingsimportexportfactory.h" ConfigOutputWidget::ConfigOutputWidget(QWidget *parent) : ConfigTaskWidget(parent),wasItMe(false) { m_config = new Ui_OutputWidget(); m_config->setupUi(this); ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); setupButtons(m_config->saveRCOutputToRAM,m_config->saveRCOutputToSD); addUAVObject("ActuatorSettings"); UAVSettingsImportExportFactory * importexportplugin = pm->getObject(); connect(importexportplugin,SIGNAL(importAboutToBegin()),this,SLOT(stopTests())); setupButtons(m_config->saveRCOutputToRAM,m_config->saveRCOutputToSD); addUAVObject("ActuatorSettings"); // First of all, put all the channel widgets into lists, so that we can // manipulate those: // NOTE: for historical reasons, we have objects below called ch0 to ch7, but the convention for OP is Channel 1 to Channel 8. outLabels << m_config->ch0OutValue << m_config->ch1OutValue << m_config->ch2OutValue << m_config->ch3OutValue << m_config->ch4OutValue << m_config->ch5OutValue << m_config->ch6OutValue << m_config->ch7OutValue << m_config->ch8OutValue << m_config->ch9OutValue; outSliders << m_config->ch0OutSlider << m_config->ch1OutSlider << m_config->ch2OutSlider << m_config->ch3OutSlider << m_config->ch4OutSlider << m_config->ch5OutSlider << m_config->ch6OutSlider << m_config->ch7OutSlider << m_config->ch8OutSlider << m_config->ch9OutSlider; outMin << m_config->ch0OutMin << m_config->ch1OutMin << m_config->ch2OutMin << m_config->ch3OutMin << m_config->ch4OutMin << m_config->ch5OutMin << m_config->ch6OutMin << m_config->ch7OutMin << m_config->ch8OutMin << m_config->ch9OutMin; outMax << m_config->ch0OutMax << m_config->ch1OutMax << m_config->ch2OutMax << m_config->ch3OutMax << m_config->ch4OutMax << m_config->ch5OutMax << m_config->ch6OutMax << m_config->ch7OutMax << m_config->ch8OutMax << m_config->ch9OutMax; reversals << m_config->ch0Rev << m_config->ch1Rev << m_config->ch2Rev << m_config->ch3Rev << m_config->ch4Rev << m_config->ch5Rev << m_config->ch6Rev << m_config->ch7Rev << m_config->ch8Rev << m_config->ch9Rev; links << m_config->ch0Link << m_config->ch1Link << m_config->ch2Link << m_config->ch3Link << m_config->ch4Link << m_config->ch5Link << m_config->ch6Link << m_config->ch7Link << m_config->ch8Link << m_config->ch9Link; // Register for ActuatorSettings changes: for (int i = 0; i < ActuatorCommand::CHANNEL_NUMELEM; i++) { connect(outMin[i], SIGNAL(editingFinished()), this, SLOT(setChOutRange())); connect(outMax[i], SIGNAL(editingFinished()), this, SLOT(setChOutRange())); connect(reversals[i], SIGNAL(toggled(bool)), this, SLOT(reverseChannel(bool))); // Now connect the channel out sliders to our signal to send updates in test mode connect(outSliders[i], SIGNAL(valueChanged(int)), this, SLOT(sendChannelTest(int))); addWidget(outMin[i]); addWidget(outMax[i]); addWidget(reversals[i]); addWidget(outSliders[i]); addWidget(links[i]); } connect(m_config->channelOutTest, SIGNAL(toggled(bool)), this, SLOT(runChannelTests(bool))); for (int i = 0; i < links.count(); i++) links[i]->setChecked(false); for (int i = 0; i < links.count(); i++) connect(links[i], SIGNAL(toggled(bool)), this, SLOT(linkToggled(bool))); refreshWidgetsValues(); firstUpdate = true; connect(m_config->spinningArmed, SIGNAL(toggled(bool)), this, SLOT(setSpinningArmed(bool))); // Connect the help button connect(m_config->outputHelp, SIGNAL(clicked()), this, SLOT(openHelp())); addWidget(m_config->outputRate3); addWidget(m_config->outputRate2); addWidget(m_config->outputRate1); addWidget(m_config->spinningArmed); UAVObjectManager *objManager = pm->getObject(); UAVObject* obj = objManager->getObject(QString("ActuatorCommand")); if(obj->getMetadata().gcsTelemetryUpdateMode == UAVObject::UPDATEMODE_ONCHANGE) this->setEnabled(false); connect(obj,SIGNAL(objectUpdated(UAVObject*)),this,SLOT(disableIfNotMe(UAVObject*))); } ConfigOutputWidget::~ConfigOutputWidget() { // Do nothing } // ************************************ /** Toggles the channel linked state for use in testing mode */ void ConfigOutputWidget::linkToggled(bool state) { Q_UNUSED(state) // find the minimum slider value for the linked ones int min = 10000; int linked_count = 0; for (int i = 0; i < outSliders.count(); i++) { if (!links[i]->checkState()) continue; int value = outSliders[i]->value(); if (min > value) min = value; linked_count++; } if (linked_count <= 0) return; // no linked channels if (!m_config->channelOutTest->checkState()) return; // we are not in Test Output mode // set the linked channels to the same value for (int i = 0; i < outSliders.count(); i++) { if (!links[i]->checkState()) continue; outSliders[i]->setValue(min); } } /** Toggles the channel testing mode by making the GCS take over the ActuatorCommand objects */ void ConfigOutputWidget::runChannelTests(bool state) { SystemAlarms * systemAlarmsObj = SystemAlarms::GetInstance(getObjectManager()); SystemAlarms::DataFields systemAlarms = systemAlarmsObj->getData(); if(state && systemAlarms.Alarm[SystemAlarms::ALARM_ACTUATOR] != SystemAlarms::ALARM_OK) { QMessageBox mbox; mbox.setText(QString(tr("The actuator module is in an error state. This can also occur because there are no inputs. Please fix these before testing outputs."))); mbox.setStandardButtons(QMessageBox::Ok); mbox.exec(); // Unfortunately must cache this since callback will reoccur accInitialData = ActuatorCommand::GetInstance(getObjectManager())->getMetadata(); m_config->channelOutTest->setChecked(false); return; } // Confirm this is definitely what they want if(state) { QMessageBox mbox; mbox.setText(QString(tr("This option will start your motors by the amount selected on the sliders regardless of transmitter. It is recommended to remove any blades from motors. Are you sure you want to do this?"))); mbox.setStandardButtons(QMessageBox::Yes | QMessageBox::No); int retval = mbox.exec(); if(retval != QMessageBox::Yes) { state = false; qDebug() << "Cancelled"; m_config->channelOutTest->setChecked(false); return; } } ActuatorCommand * obj = ActuatorCommand::GetInstance(getObjectManager()); UAVObject::Metadata mdata = obj->getMetadata(); if (state) { wasItMe=true; accInitialData = mdata; mdata.flightAccess = UAVObject::ACCESS_READONLY; mdata.flightTelemetryUpdateMode = UAVObject::UPDATEMODE_ONCHANGE; mdata.gcsTelemetryAcked = false; mdata.gcsTelemetryUpdateMode = UAVObject::UPDATEMODE_ONCHANGE; mdata.gcsTelemetryUpdatePeriod = 100; // Prevent stupid users from touching the minimum & maximum ranges while // moving the sliders. Thanks Ivan for the tip :) foreach (QSpinBox* box, outMin) { box->setEnabled(false); } foreach (QSpinBox* box, outMax) { box->setEnabled(false); } foreach (QCheckBox* box, reversals) { box->setEnabled(false); } } else { wasItMe=false; mdata = accInitialData; // Restore metadata foreach (QSpinBox* box, outMin) { box->setEnabled(true); } foreach (QSpinBox* box, outMax) { box->setEnabled(true); } foreach (QCheckBox* box, reversals) { box->setEnabled(true); } } obj->setMetadata(mdata); obj->updated(); } /** * Set the label for a channel output assignement */ void ConfigOutputWidget::assignOutputChannel(UAVDataObject *obj, QString str) { UAVObjectField* field = obj->getField(str); QStringList options = field->getOptions(); switch (options.indexOf(field->getValue().toString())) { case 0: m_config->ch0Output->setText(str); break; case 1: m_config->ch1Output->setText(str); break; case 2: m_config->ch2Output->setText(str); break; case 3: m_config->ch3Output->setText(str); break; case 4: m_config->ch4Output->setText(str); break; case 5: m_config->ch5Output->setText(str); break; case 6: m_config->ch6Output->setText(str); break; case 7: m_config->ch7Output->setText(str); break; case 8: m_config->ch8Output->setText(str); break; case 9: m_config->ch9Output->setText(str); break; } } /** * Set the "Spin motors at neutral when armed" flag in ActuatorSettings */ void ConfigOutputWidget::setSpinningArmed(bool val) { UAVDataObject *obj = dynamic_cast(getObjectManager()->getObject(QString("ActuatorSettings"))); if (!obj) return; UAVObjectField *field = obj->getField("MotorsSpinWhileArmed"); if (!field) return; if(val) field->setValue("TRUE"); else field->setValue("FALSE"); } /** Sends the channel value to the UAV to move the servo. Returns immediately if we are not in testing mode */ void ConfigOutputWidget::sendChannelTest(int value) { int in_value = value; QSlider *ob = (QSlider *)QObject::sender(); if (!ob) return; int index = outSliders.indexOf(ob); if (index < 0) return; if (reversals[index]->isChecked()) value = outMin[index]->value() - value + outMax[index]->value(); // the chsnnel is reversed // update the label outLabels[index]->setText(QString::number(value)); if (links[index]->checkState()) { // the channel is linked to other channels // set the linked channels to the same value for (int i = 0; i < outSliders.count(); i++) { if (i == index) continue; if (!links[i]->checkState()) continue; int val = in_value; if (val < outSliders[i]->minimum()) val = outSliders[i]->minimum(); if (val > outSliders[i]->maximum()) val = outSliders[i]->maximum(); if (outSliders[i]->value() == val) continue; outSliders[i]->setValue(val); outLabels[i]->setText(QString::number(val)); } } if (!m_config->channelOutTest->isChecked()) return; UAVDataObject *obj = dynamic_cast(getObjectManager()->getObject(QString("ActuatorCommand"))); if (!obj) return; UAVObjectField *channel = obj->getField("Channel"); if (!channel) return; channel->setValue(value, index); obj->updated(); } /******************************** * Output settings *******************************/ /** Request the current config from the board (RC Output) */ void ConfigOutputWidget::refreshWidgetsValues() { bool dirty=isDirty(); ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager *objManager = pm->getObject(); // Reset all channel assignements: for (int i = 0; i < ActuatorCommand::CHANNEL_NUMELEM; i++) outLabels[i]->setText("-"); // Get the channel assignements: UAVDataObject * obj = dynamic_cast(objManager->getObject(QString("ActuatorSettings"))); Q_ASSERT(obj); QList fieldList = obj->getFields(); foreach (UAVObjectField* field, fieldList) { if (field->getUnits().contains("channel")) { assignOutputChannel(obj,field->getName()); } } // Get the SpinWhileArmed setting UAVObjectField *field = obj->getField(QString("MotorsSpinWhileArmed")); m_config->spinningArmed->setChecked(field->getValue().toString().contains("TRUE")); // Get Output rates for both banks field = obj->getField(QString("ChannelUpdateFreq")); UAVObjectUtilManager* utilMngr = pm->getObject(); m_config->outputRate1->setValue(field->getValue(0).toInt()); m_config->outputRate2->setValue(field->getValue(1).toInt()); if (utilMngr) { int board = utilMngr->getBoardModel(); if ((board & 0xff00) == 1024) { // CopterControl family m_config->chBank1->setText("1-3"); m_config->chBank2->setText("4"); m_config->chBank3->setText("5,7-8"); m_config->chBank4->setText("6,9-10"); m_config->outputRate1->setEnabled(true); m_config->outputRate2->setEnabled(true); m_config->outputRate3->setEnabled(true); m_config->outputRate4->setEnabled(true); m_config->outputRate3->setValue(field->getValue(2).toInt()); m_config->outputRate4->setValue(field->getValue(3).toInt()); } else if ((board & 0xff00) == 256 ) { // Mainboard family m_config->outputRate1->setEnabled(true); m_config->outputRate2->setEnabled(true); m_config->outputRate3->setEnabled(false); m_config->outputRate4->setEnabled(false); m_config->chBank1->setText("1-4"); m_config->chBank2->setText("5-8"); m_config->chBank3->setText("-"); m_config->chBank4->setText("-"); m_config->outputRate3->setValue(0); m_config->outputRate4->setValue(0); } } // Get Channel ranges: for (int i=0;igetField(QString("ChannelMin")); int minValue = field->getValue(i).toInt(); outMin[i]->setValue(minValue); field = obj->getField(QString("ChannelMax")); int maxValue = field->getValue(i).toInt(); outMax[i]->setValue(maxValue); if (maxValue>minValue) { outSliders[i]->setMinimum(minValue); outSliders[i]->setMaximum(maxValue); reversals[i]->setChecked(false); } else { outSliders[i]->setMinimum(maxValue); outSliders[i]->setMaximum(minValue); reversals[i]->setChecked(true); } } field = obj->getField(QString("ChannelNeutral")); for (int i=0; igetValue(i).toInt(); outSliders[i]->setValue(value); outLabels[i]->setText(QString::number(value)); } setDirty(dirty); } /** * Sends the config to the board, without saving to the SD card (RC Output) */ void ConfigOutputWidget::updateObjectsFromWidgets() { ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager *objManager = pm->getObject(); UAVDataObject* obj = dynamic_cast(objManager->getObject(QString("ActuatorSettings"))); Q_ASSERT(obj); // Now send channel ranges: UAVObjectField * field = obj->getField(QString("ChannelMax")); for (int i = 0; i < ActuatorCommand::CHANNEL_NUMELEM; i++) { field->setValue(outMax[i]->value(),i); } field = obj->getField(QString("ChannelMin")); for (int i = 0; i < ActuatorCommand::CHANNEL_NUMELEM; i++) { field->setValue(outMin[i]->value(),i); } field = obj->getField(QString("ChannelNeutral")); for (int i = 0; i < ActuatorCommand::CHANNEL_NUMELEM; i++) { field->setValue(outSliders[i]->value(),i); } field = obj->getField(QString("ChannelUpdateFreq")); field->setValue(m_config->outputRate1->value(),0); field->setValue(m_config->outputRate2->value(),1); field->setValue(m_config->outputRate3->value(),2); field->setValue(m_config->outputRate4->value(),3); } /** Sets the minimum/maximum value of the channel 0 to seven output sliders. Have to do it here because setMinimum is not a slot. One added trick: if the slider is at its min when the value is changed, then keep it on the min. */ void ConfigOutputWidget::setChOutRange() { QSpinBox *spinbox = (QSpinBox*)QObject::sender(); int index = outMin.indexOf(spinbox); // This is the channel number if (index < 0) index = outMax.indexOf(spinbox); // We can't know if the signal came from min or max QSlider *slider = outSliders[index]; int oldMini = slider->minimum(); // int oldMaxi = slider->maximum(); if (outMin[index]->value()value()) { slider->setRange(outMin[index]->value(), outMax[index]->value()); reversals[index]->setChecked(false); } else { slider->setRange(outMax[index]->value(), outMin[index]->value()); reversals[index]->setChecked(true); } if (slider->value() == oldMini) slider->setValue(slider->minimum()); // if (slider->value() == oldMaxi) // slider->setValue(slider->maximum()); // this can be dangerous if it happens to be controlling a motor at the time! } /** Reverses the channel when the checkbox is clicked */ void ConfigOutputWidget::reverseChannel(bool state) { QCheckBox *checkbox = (QCheckBox*)QObject::sender(); int index = reversals.indexOf(checkbox); // This is the channel number // Sanity check: if state became true, make sure the Maxvalue was higher than Minvalue // the situations below can happen! if (state && (outMax[index]->value()value())) return; if (!state && (outMax[index]->value()>outMin[index]->value())) return; // Now, swap the min & max values (only on the spinboxes, the slider // does not change! int temp = outMax[index]->value(); outMax[index]->setValue(outMin[index]->value()); outMin[index]->setValue(temp); // Also update the channel value // This is a trick to force the slider to update its value and // emit the right signal itself, because our sendChannelTest(int) method // relies on the object sender's identity. if (outSliders[index]->value()maximum()) { outSliders[index]->setValue(outSliders[index]->value()+1); outSliders[index]->setValue(outSliders[index]->value()-1); } else { outSliders[index]->setValue(outSliders[index]->value()-1); outSliders[index]->setValue(outSliders[index]->value()+1); } } void ConfigOutputWidget::openHelp() { QDesktopServices::openUrl( QUrl("http://wiki.openpilot.org/display/Doc/Output+Configuration", QUrl::StrictMode) ); } void ConfigOutputWidget::stopTests() { m_config->channelOutTest->setChecked(false); } void ConfigOutputWidget::disableIfNotMe(UAVObject* obj) { if(obj->getMetadata().gcsTelemetryUpdateMode == UAVObject::UPDATEMODE_ONCHANGE) { if(!wasItMe) this->setEnabled(false); } else this->setEnabled(true); }