/** ****************************************************************************** * * @file configtaskwidget.cpp * @author E. Lafargue & The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010. * @addtogroup GCSPlugins GCS Plugins * @{ * @addtogroup ConfigPlugin Config Plugin * @{ * @brief The Configuration Gadget used to update settings in the firmware *****************************************************************************/ /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "configtaskwidget.h" #include #include "uavsettingsimportexport/uavsettingsimportexportfactory.h" #include "configgadgetwidget.h" ConfigTaskWidget::ConfigTaskWidget(QWidget *parent) : QWidget(parent),isConnected(false),smartsave(NULL),dirty(false),outOfLimitsStyle("background-color: rgb(255, 0, 0);") { pm = ExtensionSystem::PluginManager::instance(); objManager = pm->getObject(); connect((ConfigGadgetWidget*)parent, SIGNAL(autopilotConnected()),this, SLOT(onAutopilotConnect())); connect((ConfigGadgetWidget*)parent, SIGNAL(autopilotDisconnected()),this, SLOT(onAutopilotDisconnect())); UAVSettingsImportExportFactory * importexportplugin = pm->getObject(); connect(importexportplugin,SIGNAL(importAboutToBegin()),this,SLOT(invalidateObjects())); } void ConfigTaskWidget::addWidget(QWidget * widget) { addUAVObjectToWidgetRelation("","",widget); } void ConfigTaskWidget::addUAVObject(QString objectName) { addUAVObjectToWidgetRelation(objectName,"",NULL); } void ConfigTaskWidget::addUAVObjectToWidgetRelation(QString object, QString field, QWidget * widget, QString index) { UAVObject *obj=NULL; UAVObjectField *_field=NULL; obj = objManager->getObject(QString(object)); Q_ASSERT(obj); _field = obj->getField(QString(field)); Q_ASSERT(_field); addUAVObjectToWidgetRelation(object,field,widget,_field->getElementNames().indexOf(index)); } void ConfigTaskWidget::addUAVObjectToWidgetRelation(QString object, QString field, QWidget *widget, QString element, float scale, bool isLimited, QList *defaultReloadGroups) { UAVObject *obj=objManager->getObject(QString(object)); Q_ASSERT(obj); UAVObjectField *_field; if(!field.isEmpty() && obj) _field = obj->getField(QString(field)); int index=_field->getElementNames().indexOf(QString(element)); addUAVObjectToWidgetRelation(object, field, widget,index,scale,isLimited,defaultReloadGroups); } void ConfigTaskWidget::addUAVObjectToWidgetRelation(QString object, QString field, QWidget * widget, int index,float scale,bool isLimited,QList* defaultReloadGroups) { if(addShadowWidget(object,field,widget,index,scale,isLimited,defaultReloadGroups)) return; UAVObject *obj=NULL; UAVObjectField *_field=NULL; if(!object.isEmpty()) { obj = objManager->getObject(QString(object)); Q_ASSERT(obj); objectUpdates.insert(obj,true); connect(obj, SIGNAL(objectUpdated(UAVObject*)),this, SLOT(objectUpdated(UAVObject*))); connect(obj, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(refreshWidgetsValues())); } if(!field.isEmpty() && obj) _field = obj->getField(QString(field)); objectToWidget * ow=new objectToWidget(); ow->field=_field; ow->object=obj; ow->widget=widget; ow->index=index; ow->scale=scale; ow->isLimited=isLimited; objOfInterest.append(ow); if(obj) { if(smartsave) { smartsave->addObject((UAVDataObject*)obj); emit objectAdded(obj); } } if(widget==NULL) { // do nothing } else { connectWidgetUpdatesToSlot(widget,SLOT(widgetsContentsChanged())); if(defaultReloadGroups) addWidgetToDefaultReloadGroups(widget,defaultReloadGroups); shadowsList.insert(widget,ow); loadWidgetLimits(widget,_field,index,isLimited,scale); } } ConfigTaskWidget::~ConfigTaskWidget() { if(smartsave) delete smartsave; foreach(QList* pointer,defaultReloadGroups.values()) { if(pointer) delete pointer; } foreach (objectToWidget* oTw, objOfInterest) { if(oTw) delete oTw; } } void ConfigTaskWidget::saveObjectToSD(UAVObject *obj) { qDebug()<<"ConfigTaskWidget::saveObjectToSD"; // saveObjectToSD is now handled by the UAVUtils plugin in one // central place (and one central queue) ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectUtilManager* utilMngr = pm->getObject(); utilMngr->saveObjectToSD(obj); } UAVObjectManager* ConfigTaskWidget::getObjectManager() { ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance(); UAVObjectManager * objMngr = pm->getObject(); Q_ASSERT(objMngr); return objMngr; } double ConfigTaskWidget::listMean(QList list) { double accum = 0; for(int i = 0; i < list.size(); i++) accum += list[i]; return accum / list.size(); } // ************************************ // telemetry start/stop connect/disconnect signals void ConfigTaskWidget::onAutopilotDisconnect() { isConnected=false; enableControls(false); invalidateObjects(); } void ConfigTaskWidget::onAutopilotConnect() { invalidateObjects(); dirty=false; isConnected=true; enableControls(true); refreshWidgetsValues(); } void ConfigTaskWidget::populateWidgets() { bool dirtyBack=dirty; foreach(objectToWidget * ow,objOfInterest) { if(ow->object==NULL || ow->field==NULL || ow->widget==NULL) { // do nothing } else setWidgetFromField(ow->widget,ow->field,ow->index,ow->scale,ow->isLimited); } setDirty(dirtyBack); } void ConfigTaskWidget::refreshWidgetsValues() { bool dirtyBack=dirty; foreach(objectToWidget * ow,objOfInterest) { if(ow->object==NULL || ow->field==NULL || ow->widget==NULL) { //do nothing } else { setWidgetFromField(ow->widget,ow->field,ow->index,ow->scale,ow->isLimited); } } setDirty(dirtyBack); } void ConfigTaskWidget::updateObjectsFromWidgets() { foreach(objectToWidget * ow,objOfInterest) { if(ow->object==NULL || ow->field==NULL) { } else setFieldFromWidget(ow->widget,ow->field,ow->index,ow->scale); } } void ConfigTaskWidget::addApplySaveButtons(QPushButton *update, QPushButton *save) { smartsave=new smartSaveButton(update,save); connect(smartsave,SIGNAL(preProcessOperations()), this, SLOT(updateObjectsFromWidgets())); connect(smartsave,SIGNAL(saveSuccessfull()),this,SLOT(clearDirty())); connect(smartsave,SIGNAL(beginOp()),this,SLOT(disableObjUpdates())); connect(smartsave,SIGNAL(endOp()),this,SLOT(enableObjUpdates())); if(objOfInterest.count()>0) { foreach(objectToWidget * oTw,objOfInterest) { smartsave->addObject((UAVDataObject*)oTw->object); emit objectAdded(oTw->object); } } enableControls(false); } void ConfigTaskWidget::enableControls(bool enable) { if(smartsave) smartsave->enableControls(enable); } void ConfigTaskWidget::widgetsContentsChanged() { float scale; objectToWidget * oTw= shadowsList.value((QWidget*)sender(),NULL); /* qDebug()<<"sender:"<<(quint32)sender(); foreach(QWidget * w,shadowsList.keys()) qDebug()<<"in list:"<<(quint32)w; if(oTw) qDebug()<<"in oTw OK"<<(quint32)oTw->widget; */ if(oTw) { if(oTw->widget==(QWidget*)sender()) { scale=oTw->scale; checkWidgetsLimits((QWidget*)sender(),oTw->field,oTw->index,oTw->isLimited,getVariantFromWidget((QWidget*)sender(),oTw->scale),oTw->scale); qDebug()<<"sender was master"; } else { foreach (shadow * sh, oTw->shadowsList) { if(sh->widget==(QWidget*)sender()) { scale=sh->scale; checkWidgetsLimits((QWidget*)sender(),oTw->field,oTw->index,sh->isLimited,getVariantFromWidget((QWidget*)sender(),scale),scale); qDebug()<<"sender was shadow"; } } } if(oTw->widget!=(QWidget *)sender()) { checkWidgetsLimits(oTw->widget,oTw->field,oTw->index,oTw->isLimited,getVariantFromWidget((QWidget*)sender(),scale),oTw->scale); setWidgetFromVariant(oTw->widget,getVariantFromWidget((QWidget*)sender(),scale),oTw->scale); } foreach (shadow * sh, oTw->shadowsList) { if(sh->widget!=(QWidget*)sender()) { checkWidgetsLimits(sh->widget,oTw->field,oTw->index,sh->isLimited,getVariantFromWidget((QWidget*)sender(),scale),oTw->scale); setWidgetFromVariant(sh->widget,getVariantFromWidget((QWidget*)sender(),scale),sh->scale); } } } setDirty(true); } void ConfigTaskWidget::clearDirty() { setDirty(false); } void ConfigTaskWidget::setDirty(bool value) { dirty=value; } bool ConfigTaskWidget::isDirty() { if(isConnected) return dirty; else return false; } void ConfigTaskWidget::disableObjUpdates() { foreach(objectToWidget * obj,objOfInterest) { if(obj->object) disconnect(obj->object, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(refreshWidgetsValues())); } } void ConfigTaskWidget::enableObjUpdates() { foreach(objectToWidget * obj,objOfInterest) { if(obj->object) connect(obj->object, SIGNAL(objectUpdated(UAVObject*)), this, SLOT(refreshWidgetsValues())); } } void ConfigTaskWidget::objectUpdated(UAVObject *obj) { objectUpdates[obj]=true; } void ConfigTaskWidget::removeObject(UAVObject * obj) { emit objectRemoved(obj); QList toRemove; foreach(objectToWidget * oTw,objOfInterest) { if(oTw->object==obj) { toRemove.append(oTw); } } foreach(objectToWidget * oTw,toRemove) { objOfInterest.removeAll(oTw); smartsave->removeObject((UAVDataObject*)oTw->object); } } void ConfigTaskWidget::removeAllObjects() { foreach(objectToWidget * oTw,objOfInterest) { emit objectRemoved(oTw->object); } objOfInterest.clear(); smartsave->removeAllObjects(); } bool ConfigTaskWidget::allObjectsUpdated() { bool ret=true; foreach(UAVObject *obj, objectUpdates.keys()) { ret=ret & objectUpdates[obj]; } qDebug()<<"ALL OBJECTS UPDATE:"<* defaultReloadGroups) { foreach(objectToWidget * oTw,objOfInterest) { if(!oTw->object || !oTw->widget) continue; if(oTw->object->getName()==object && oTw->field->getName()==field && oTw->index==index) { shadow * sh=new shadow; sh->isLimited=isLimited; sh->scale=scale; sh->widget=widget; oTw->shadowsList.append(sh); shadowsList.insert(widget,oTw); connectWidgetUpdatesToSlot(widget,SLOT(widgetsContentsChanged())); if(defaultReloadGroups) addWidgetToDefaultReloadGroups(widget,defaultReloadGroups); return true; } } return false; } void ConfigTaskWidget::autoLoadWidgets() { QPushButton * saveButtonWidget=NULL; QPushButton * applyButtonWidget=NULL; foreach(QObject * widget,this->children()) { QVariant info=widget->property("objrelation"); if(info.isValid()) { uiRelationAutomation uiRelation; uiRelation.buttonType=none; foreach(QString str, info.toStringList()) { QString prop=str.split(":").at(0); QString value=str.split(":").at(1); if(prop== "objname") uiRelation.objname=value; else if(prop== "fieldname") uiRelation.fieldname=value; else if(prop=="element") uiRelation.element=value; else if(prop== "scale") { if(value=="null") uiRelation.scale=1; else uiRelation.scale=value.toFloat(); } else if(prop== "ismaster") { if(value=="yes") uiRelation.ismaster=true; else uiRelation.ismaster=false; } else if(prop== "haslimits") { if(value=="yes") uiRelation.haslimits=true; else uiRelation.haslimits=false; } else if(prop== "button") { if(value=="save") uiRelation.buttonType=save_button; else if(value=="apply") uiRelation.buttonType=apply_button; else if(value=="reload") uiRelation.buttonType=reload_button; else if(value=="default") uiRelation.buttonType=default_button; } else if(prop== "buttongroup") { foreach(QString s,value.split(",")) { uiRelation.buttonGroup.append(s.toInt()); } } } if(!uiRelation.buttonType==none) { QPushButton * button=NULL; switch(uiRelation.buttonType) { case save_button: saveButtonWidget=qobject_cast(widget); break; case apply_button: applyButtonWidget=qobject_cast(widget); break; case default_button: button=qobject_cast(widget); if(button) addDefaultButton(button,uiRelation.buttonGroup.at(0)); break; case reload_button: button=qobject_cast(widget); if(button) addReloadButton(button,uiRelation.buttonGroup.at(0)); break; default: break; } } else if(uiRelation.ismaster) { QWidget * wid=qobject_cast(widget); if(wid) addUAVObjectToWidgetRelation(uiRelation.objname,uiRelation.fieldname,wid,uiRelation.element,uiRelation.haslimits,&uiRelation.buttonGroup); } } } if(saveButtonWidget && applyButtonWidget) addApplySaveButtons(applyButtonWidget,saveButtonWidget); } void ConfigTaskWidget::addWidgetToDefaultReloadGroups(QWidget *widget, QList * groups) { foreach(objectToWidget * oTw,objOfInterest) { bool addOTW=false; if(oTw->widget==widget) addOTW=true; else { foreach(shadow * sh, oTw->shadowsList) { if(sh->widget==widget) addOTW=true; } } if(addOTW) { foreach(int i,*groups) { if(defaultReloadGroups.contains(i)) { defaultReloadGroups.value(i)->append(oTw); } else { defaultReloadGroups.insert(i,new QList()); defaultReloadGroups.value(i)->append(oTw); } } } } } void ConfigTaskWidget::addDefaultButton(QPushButton *button, int buttonGroup) { button->setProperty("group",buttonGroup); connect(button,SIGNAL(clicked()),this,SLOT(defaultButtonClicked())); } void ConfigTaskWidget::addReloadButton(QPushButton *button, int buttonGroup) { button->setProperty("group",buttonGroup); connect(button,SIGNAL(clicked()),this,SLOT(reloadButtonClicked())); } void ConfigTaskWidget::defaultButtonClicked() { int group=sender()->property("group").toInt(); QList * list=defaultReloadGroups.value(group); foreach(objectToWidget * oTw,*list) { if(!oTw->object) continue; UAVDataObject * temp=((UAVDataObject*)oTw->object)->dirtyClone(); setWidgetFromField(oTw->widget,temp->getField(oTw->field->getName()),oTw->index,oTw->scale,oTw->isLimited); } } void ConfigTaskWidget::reloadButtonClicked() { int group=sender()->property("group").toInt(); QList * list=defaultReloadGroups.value(group); ObjectPersistence* objper = dynamic_cast( getObjectManager()->getObject(ObjectPersistence::NAME) ); QTimer * timeOut=new QTimer(this); QEventLoop * eventLoop=new QEventLoop(this); connect(timeOut, SIGNAL(timeout()),eventLoop,SLOT(quit())); connect(objper, SIGNAL(objectUpdated(UAVObject*)), eventLoop, SLOT(quit())); foreach(objectToWidget * oTw,*list) { if (oTw->object != NULL) { ObjectPersistence::DataFields data; data.Operation = ObjectPersistence::OPERATION_LOAD; data.Selection = ObjectPersistence::SELECTION_SINGLEOBJECT; data.ObjectID = oTw->object->getObjID(); data.InstanceID = oTw->object->getInstID(); objper->setData(data); objper->updated(); timeOut->start(500); eventLoop->exec(); if(timeOut->isActive()) { setWidgetFromField(oTw->widget,oTw->field,oTw->index,oTw->scale,oTw->isLimited); } timeOut->stop(); } } delete eventLoop; delete timeOut; } void ConfigTaskWidget::connectWidgetUpdatesToSlot(QWidget * widget,const char* function) { if(!widget) return; if(QComboBox * cb=qobject_cast(widget)) { connect(cb,SIGNAL(currentIndexChanged(int)),this,function); } else if(QSlider * cb=qobject_cast(widget)) { connect(cb,SIGNAL(sliderMoved(int)),this,function); } else if(MixerCurveWidget * cb=qobject_cast(widget)) { connect(cb,SIGNAL(curveUpdated(QList,double)),this,function); } else if(QTableWidget * cb=qobject_cast(widget)) { connect(cb,SIGNAL(cellChanged(int,int)),this,function); } else if(QSpinBox * cb=qobject_cast(widget)) { connect(cb,SIGNAL(valueChanged(int)),this,function); } else if(QDoubleSpinBox * cb=qobject_cast(widget)) { connect(cb,SIGNAL(valueChanged(double)),this,function); } else if(QCheckBox * cb=qobject_cast(widget)) { connect(cb,SIGNAL(clicked()),this,function); } else if(QPushButton * cb=qobject_cast(widget)) { connect(cb,SIGNAL(clicked()),this,function); } else qDebug()<<__FUNCTION__<<"widget to uavobject relation not implemented"<metaObject()->className(); } bool ConfigTaskWidget::setFieldFromWidget(QWidget * widget,UAVObjectField * field,int index,float scale) { if(!widget || !field) return false; QVariant ret=getVariantFromWidget(widget,scale); if(ret.isValid()) { field->setValue(ret,index); return true; } { qDebug()<<__FUNCTION__<<"widget to uavobject relation not implemented"<metaObject()->className(); return false; } } QVariant ConfigTaskWidget::getVariantFromWidget(QWidget * widget,float scale) { if(QComboBox * cb=qobject_cast(widget)) { return (QString)cb->currentText(); } else if(QLabel * cb=qobject_cast(widget)) { return (QString)cb->text(); } else if(QDoubleSpinBox * cb=qobject_cast(widget)) { return (double)(cb->value()* scale); } else if(QSpinBox * cb=qobject_cast(widget)) { return (int)(cb->value()* (int)scale); } else if(QSlider * cb=qobject_cast(widget)) { return(int)(cb->value()* (int)scale); } else if(QCheckBox * cb=qobject_cast(widget)) { return (bool)(cb->isChecked()?"TRUE":"FALSE"); } else return QVariant(); } bool ConfigTaskWidget::setWidgetFromVariant(QWidget *widget, QVariant value, float scale) { if(QComboBox * cb=qobject_cast(widget)) { cb->setCurrentIndex(cb->findText(value.toString())); return true; } else if(QLabel * cb=qobject_cast(widget)) { cb->setText(value.toString()); return true; } else if(QDoubleSpinBox * cb=qobject_cast(widget)) { cb->setValue(value.toDouble()/scale); return true; } else if(QSpinBox * cb=qobject_cast(widget)) { cb->setValue(value.toInt()/(int)scale); return true; } else if(QSlider * cb=qobject_cast(widget)) { cb->setValue(value.toInt()/(int)scale); return true; } else if(QCheckBox * cb=qobject_cast(widget)) { cb->setChecked(value.toBool()); return true; } else return false; } bool ConfigTaskWidget::setWidgetFromField(QWidget * widget,UAVObjectField * field,int index,float scale,bool hasLimits) { if(!widget || !field) return false; if(QComboBox * cb=qobject_cast(widget)) { if(cb->count()==0) loadWidgetLimits(cb,field,index,hasLimits,scale); } QVariant var=field->getValue(index); checkWidgetsLimits(widget,field,index,hasLimits,var,scale); bool ret=setWidgetFromVariant(widget,var,scale); if(ret) return true; else { qDebug()<<__FUNCTION__<<"widget to uavobject relation not implemented"<metaObject()->className(); return false; } } void ConfigTaskWidget::checkWidgetsLimits(QWidget * widget,UAVObjectField * field,int index,bool hasLimits, QVariant value, float scale) { if(!hasLimits) return; qDebug()<<"check widget"<accessibleName()<<"value"<isWithinLimits(value,index); if(!field->isWithinLimits(value,index)) { if(!widget->property("styleBackup").isValid()) widget->setProperty("styleBackup",widget->styleSheet()); widget->setStyleSheet(outOfLimitsStyle); widget->setProperty("wasOverLimits",(bool)true); if(QComboBox * cb=qobject_cast(widget)) { if(cb->findText(value.toString())==-1) cb->addItem(value.toString()); } else if(QDoubleSpinBox * cb=qobject_cast(widget)) { if(value.toDouble()/scale>cb->maximum()) { cb->setMaximum(value.toDouble()/scale); } else if(value.toDouble()/scaleminimum()) { cb->setMinimum(value.toDouble()/scale); } } else if(QSpinBox * cb=qobject_cast(widget)) { if(value.toInt()/scale>cb->maximum()) { cb->setMaximum(value.toInt()/scale); } else if(value.toInt()/scaleminimum()) { cb->setMinimum(value.toInt()/scale); } } else if(QSlider * cb=qobject_cast(widget)) { if(value.toInt()/scale>cb->maximum()) { cb->setMaximum(value.toInt()/scale); } else if(value.toInt()/scaleminimum()) { cb->setMinimum(value.toInt()/scale); } } } else if(widget->property("wasOverLimits").isValid()) { qDebug()<<"1wasOverLimits"; if(widget->property("wasOverLimits").toBool()) { qDebug()<<"2"; widget->setProperty("wasOverLimits",(bool)false); if(widget->property("styleBackup").isValid()) { qDebug()<<"3"; QString style=widget->property("styleBackup").toString(); qDebug()<<"STYLE"<setStyleSheet(style); } loadWidgetLimits(widget,field,index,hasLimits,scale); } } } void ConfigTaskWidget::loadWidgetLimits(QWidget * widget,UAVObjectField * field,int index,bool hasLimits,float scale) { if(QComboBox * cb=qobject_cast(widget)) { cb->clear(); QStringList option=field->getOptions(); if(hasLimits) { foreach(QString str,option) { if(field->isWithinLimits(str,index)) cb->addItem(str); } } else cb->addItems(option); } if(!hasLimits) return; else if(QDoubleSpinBox * cb=qobject_cast(widget)) { if(field->getMaxLimit(index).isValid()) { cb->setMaximum(field->getMaxLimit(index).toDouble()/scale); } if(field->getMinLimit(index).isValid()) { cb->setMinimum(field->getMinLimit(index).toDouble()/scale); } } else if(QSpinBox * cb=qobject_cast(widget)) { if(field->getMaxLimit(index).isValid()) { cb->setMaximum((int)(field->getMaxLimit(index).toInt()/scale)); } if(field->getMinLimit(index).isValid()) { cb->setMinimum((int)(field->getMinLimit(index).toInt()/scale)); } } else if(QSlider * cb=qobject_cast(widget)) { if(field->getMaxLimit(index).isValid()) { cb->setMaximum((int)(field->getMaxLimit(index).toInt()/scale)); } if(field->getMinLimit(index).isValid()) { cb->setMinimum((int)(field->getMinLimit(index).toInt()/scale)); } } } /** @} @} */