/*===================================================================== QGroundControl Open Source Ground Control Station (c) 2009, 2015 QGROUNDCONTROL PROJECT This file is part of the QGROUNDCONTROL project QGROUNDCONTROL is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. QGROUNDCONTROL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with QGROUNDCONTROL. If not, see . ======================================================================*/ /// @file /// @author Don Gagne #include "AirframeComponentController.h" #include "AirframeComponentAirframes.h" #include "QGCMAVLink.h" #include "MultiVehicleManager.h" #include "AutoPilotPluginManager.h" #include "QGCApplication.h" #include "QGCMessageBox.h" #include #include bool AirframeComponentController::_typesRegistered = false; AirframeComponentController::AirframeComponentController(void) : _currentVehicleIndex(0), _autostartId(0), _showCustomConfigPanel(false) { if (!_typesRegistered) { _typesRegistered = true; qmlRegisterUncreatableType("QGroundControl.Controllers", 1, 0, "AiframeType", "Can only reference AirframeType"); qmlRegisterUncreatableType("QGroundControl.Controllers", 1, 0, "Aiframe", "Can only reference Airframe"); } QStringList usedParams; usedParams << "SYS_AUTOSTART" << "SYS_AUTOCONFIG"; if (!_allParametersExists(FactSystem::defaultComponentId, usedParams)) { return; } // Load up member variables bool autostartFound = false; _autostartId = getParameterFact(FactSystem::defaultComponentId, "SYS_AUTOSTART")->value().toInt(); for (int tindex = 0; tindex < AirframeComponentAirframes::get().count(); tindex++) { const AirframeComponentAirframes::AirframeType_t* pType = AirframeComponentAirframes::get().values().at(tindex); AirframeType* airframeType = new AirframeType(pType->name, pType->imageResource, this); Q_CHECK_PTR(airframeType); for (int index = 0; index < pType->rgAirframeInfo.count(); index++) { const AirframeComponentAirframes::AirframeInfo_t* pInfo = pType->rgAirframeInfo.at(index); Q_CHECK_PTR(pInfo); if (_autostartId == pInfo->autostartId) { Q_ASSERT(!autostartFound); autostartFound = true; _currentAirframeType = pType->name; _currentVehicleName = pInfo->name; _currentVehicleIndex = index; } airframeType->addAirframe(pInfo->name, pInfo->autostartId); } _airframeTypes.append(QVariant::fromValue(airframeType)); } if (_autostartId != 0 && !autostartFound) { _showCustomConfigPanel = true; emit showCustomConfigPanelChanged(true); } } AirframeComponentController::~AirframeComponentController() { } void AirframeComponentController::changeAutostart(void) { if (qgcApp()->toolbox()->multiVehicleManager()->vehicles()->count() > 1) { QGCMessageBox::warning("Airframe Config", "You cannot change airframe configuration while connected to multiple vehicles."); return; } qgcApp()->setOverrideCursor(Qt::WaitCursor); Fact* sysAutoStartFact = getParameterFact(-1, "SYS_AUTOSTART"); Fact* sysAutoConfigFact = getParameterFact(-1, "SYS_AUTOCONFIG"); // We need to wait for the vehicleUpdated signals to come back before we reboot _waitParamWriteSignalCount = 0; connect(sysAutoStartFact, &Fact::vehicleUpdated, this, &AirframeComponentController::_waitParamWriteSignal); connect(sysAutoConfigFact, &Fact::vehicleUpdated, this, &AirframeComponentController::_waitParamWriteSignal); // We use forceSetValue to params are sent even if the previous value is that same as the new value sysAutoStartFact->forceSetValue(_autostartId); sysAutoConfigFact->forceSetValue(1); } void AirframeComponentController::_waitParamWriteSignal(QVariant value) { Q_UNUSED(value); _waitParamWriteSignalCount++; if (_waitParamWriteSignalCount == 2) { // Now that both params have made it to the vehicle we can reboot it. All these signals are flying // around on the main thread, so we need to allow the stack to unwind back to the event loop before // we reboot. QTimer::singleShot(800, this, &AirframeComponentController::_rebootAfterStackUnwind); } } void AirframeComponentController::_rebootAfterStackUnwind(void) { _uas->executeCommand(MAV_CMD_PREFLIGHT_REBOOT_SHUTDOWN, 1, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0); qgcApp()->processEvents(QEventLoop::ExcludeUserInputEvents); for (unsigned i = 0; i < 2000; i++) { QGC::SLEEP::usleep(500); qgcApp()->processEvents(QEventLoop::ExcludeUserInputEvents); } qgcApp()->toolbox()->linkManager()->disconnectAll(); qgcApp()->restoreOverrideCursor(); } AirframeType::AirframeType(const QString& name, const QString& imageResource, QObject* parent) : QObject(parent), _name(name), _imageResource(imageResource) { } AirframeType::~AirframeType() { } void AirframeType::addAirframe(const QString& name, int autostartId) { Airframe* airframe = new Airframe(name, autostartId); Q_CHECK_PTR(airframe); _airframes.append(QVariant::fromValue(airframe)); } Airframe::Airframe(const QString& name, int autostartId, QObject* parent) : QObject(parent), _name(name), _autostartId(autostartId) { } Airframe::~Airframe() { }