/* Copyright (c) 2011-2014, Mathieu Labbe - IntRoLab - Universite de Sherbrooke All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of the Universite de Sherbrooke nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "find_object/Settings.h" #include "ParametersToolBox.h" #include #include #include #include #include #include #include #include #include namespace find_object { ParametersToolBox::ParametersToolBox(QWidget *parent) : QToolBox(parent) { } ParametersToolBox::~ParametersToolBox() { } QWidget * ParametersToolBox::getParameterWidget(const QString & key) { return this->findChild(key); } QStringList ParametersToolBox::resetPage(int index) { QStringList paramChanged; const QObjectList & children = this->widget(index)->children(); for(int j=0; jobjectName(); // ignore only the nextObjID setting, to avoid problem with saved objects if(key.compare(Settings::kGeneral_nextObjID()) != 0) { QVariant value = Settings::getDefaultParameters().value(key, QVariant()); if(value.isValid()) { Settings::setParameter(key, value); if(qobject_cast(children.at(j))) { if(((QComboBox*)children.at(j))->currentIndex() != value.toString().split(':').first().toInt()) { ((QComboBox*)children.at(j))->setCurrentIndex(value.toString().split(':').first().toInt()); paramChanged.append(key); } } else if(qobject_cast(children.at(j))) { if(((QSpinBox*)children.at(j))->value() != value.toInt()) { ((QSpinBox*)children.at(j))->setValue(value.toInt()); paramChanged.append(key); } } else if(qobject_cast(children.at(j))) { if(((QDoubleSpinBox*)children.at(j))->value() != value.toDouble()) { ((QDoubleSpinBox*)children.at(j))->setValue(value.toDouble()); paramChanged.append(key); } } else if(qobject_cast(children.at(j))) { if(((QCheckBox*)children.at(j))->isChecked() != value.toBool()) { ((QCheckBox*)children.at(j))->setChecked(value.toBool()); paramChanged.append(key); } } else if(qobject_cast(children.at(j))) { if(((QLineEdit*)children.at(j))->text().compare(value.toString()) != 0) { ((QLineEdit*)children.at(j))->setText(value.toString()); paramChanged.append(key); } } } } } return paramChanged; } void ParametersToolBox::resetCurrentPage() { this->blockSignals(true); QStringList paramChanged = this->resetPage(this->currentIndex()); this->blockSignals(false); Q_EMIT parametersChanged(paramChanged); } void ParametersToolBox::resetAllPages() { QStringList paramChanged; this->blockSignals(true); for(int i=0; i< this->count(); ++i) { paramChanged.append(this->resetPage(i)); } this->blockSignals(false); Q_EMIT parametersChanged(paramChanged); } void ParametersToolBox::updateParametersVisibility() { //show/hide not used parameters QComboBox * descriptorBox = this->findChild(Settings::kFeature2D_2Descriptor()); QComboBox * detectorBox = this->findChild(Settings::kFeature2D_1Detector()); if(descriptorBox && detectorBox) { QString group = Settings::kFeature2D_2Descriptor().split('/').first(); QWidget * panel = 0; for(int i=0; icount(); ++i) { if(this->widget(i)->objectName().compare(group) == 0) { panel = this->widget(i); break; } } if(panel) { const QObjectList & objects = panel->children(); QString descriptorName = descriptorBox->currentText(); QString detectorName = detectorBox->currentText(); for(int i=0; iobjectName().isEmpty()) { if(objects[i]->objectName().contains(descriptorName) || objects[i]->objectName().contains(detectorName)) { ((QWidget*)objects[i])->setVisible(true); } else if(!objects[i]->objectName().split('/').at(1).at(0).isDigit()) { ((QWidget*)objects[i])->setVisible(false); } } } } } QComboBox * nnBox = this->findChild(Settings::kNearestNeighbor_1Strategy()); if(nnBox) { QString group = Settings::kNearestNeighbor_1Strategy().split('/').first(); QWidget * panel = 0; for(int i=0; icount(); ++i) { if(this->widget(i)->objectName().compare(group) == 0) { panel = this->widget(i); break; } } if(panel) { const QObjectList & objects = panel->children(); QString nnName = nnBox->currentText(); for(int i=0; iobjectName().isEmpty()) { if(objects[i]->objectName().contains(nnName)) { ((QWidget*)objects[i])->setVisible(true); } else if(!objects[i]->objectName().split('/').at(1).at(0).isDigit()) { ((QWidget*)objects[i])->setVisible(false); } } } } } } void ParametersToolBox::setupUi() { this->removeItem(0); // remove dummy page used in .ui QWidget * currentItem = 0; const ParametersMap & parameters = Settings::getParameters(); for(ParametersMap::const_iterator iter=parameters.constBegin(); iter!=parameters.constEnd(); ++iter) { QStringList splitted = iter.key().split('/'); QString group = splitted.first(); QString name = splitted.last(); if(currentItem == 0 || currentItem->objectName().compare(group) != 0) { currentItem = new QWidget(this); this->addItem(currentItem, group); currentItem->setObjectName(group); QVBoxLayout * layout = new QVBoxLayout(currentItem); currentItem->setLayout(layout); layout->setContentsMargins(0,0,0,0); layout->setSpacing(0); layout->addSpacerItem(new QSpacerItem(0,0, QSizePolicy::Minimum, QSizePolicy::Expanding)); addParameter(layout, iter.key(), iter.value()); } else { addParameter((QVBoxLayout*)currentItem->layout(), iter.key(), iter.value()); } } updateParametersVisibility(); } void ParametersToolBox::updateParameter(const QString & key) { QWidget * widget = this->findChild(key); QString type = Settings::getParametersType().value(key); if(type.compare("QString") == 0) { QString value = Settings::getParameter(key).toString(); if(value.contains(';')) { // It's a list, just change the index QStringList splitted = value.split(':'); ((QComboBox*)widget)->setCurrentIndex(splitted.first().toInt()); } else { ((QLineEdit*)widget)->setText(value); } } else if(type.compare("int") == 0) { ((QSpinBox*)widget)->setValue(Settings::getParameter(key).toInt()); } else if(type.compare("uint") == 0) { ((QSpinBox*)widget)->setValue(Settings::getParameter(key).toInt()); } else if(type.compare("double") == 0) { ((QDoubleSpinBox*)widget)->setValue(Settings::getParameter(key).toDouble()); } else if(type.compare("float") == 0) { ((QDoubleSpinBox*)widget)->setValue(Settings::getParameter(key).toDouble()); } else if(type.compare("bool") == 0) { ((QCheckBox*)widget)->setChecked(Settings::getParameter(key).toBool()); } } void ParametersToolBox::addParameter(QVBoxLayout * layout, const QString & key, const QVariant & value) { QString type = Settings::getParametersType().value(key); if(type.compare("QString") == 0) { addParameter(layout, key, value.toString()); } else if(type.compare("int") == 0) { addParameter(layout, key, value.toInt()); } else if(type.compare("uint") == 0) { addParameter(layout, key, value.toInt()); } else if(type.compare("double") == 0) { addParameter(layout, key, value.toDouble()); } else if(type.compare("float") == 0) { addParameter(layout, key, value.toDouble()); } else if(type.compare("bool") == 0) { addParameter(layout, key, value.toBool()); } } void ParametersToolBox::addParameter(QVBoxLayout * layout, const QString & key, const QString & value) { if(value.contains(';')) { QComboBox * widget = new QComboBox(this); widget->setObjectName(key); QStringList splitted = value.split(':'); widget->addItems(splitted.last().split(';')); widget->setCurrentIndex(splitted.first().toInt()); connect(widget, SIGNAL(currentIndexChanged(int)), this, SLOT(changeParameter(int))); addParameter(layout, key, widget); } else { QLineEdit * widget = new QLineEdit(value, this); widget->setObjectName(key); connect(widget, SIGNAL(editingFinished()), this, SLOT(changeParameter())); addParameter(layout, key, widget); } } void ParametersToolBox::addParameter(QVBoxLayout * layout, const QString & key, const double & value) { QDoubleSpinBox * widget = new QDoubleSpinBox(this); double def = Settings::getDefaultParameters().value(key).toDouble(); if(def<0.001) { widget->setDecimals(4); } else if(def<0.01) { widget->setDecimals(3); } if(def>=0.0) { widget->setMaximum(def*1000000.0); } else if(def==0.0) { widget->setMaximum(1000000.0); } else if(def<0.0) { widget->setMinimum(def*1000000.0); widget->setMaximum(0.0); } widget->setValue(value); widget->setObjectName(key); connect(widget, SIGNAL(editingFinished()), this, SLOT(changeParameter())); addParameter(layout, key, widget); } void ParametersToolBox::addParameter(QVBoxLayout * layout, const QString & key, const int & value) { QSpinBox * widget = new QSpinBox(this); int def = Settings::getDefaultParameters().value(key).toInt(); if(def>0) { widget->setMaximum(def*1000000); } else if(def == 0) { widget->setMaximum(1000000); } else if(def<0) { widget->setMinimum(def*1000000); widget->setMaximum(0); } widget->setValue(value); widget->setObjectName(key); connect(widget, SIGNAL(editingFinished()), this, SLOT(changeParameter())); addParameter(layout, key, widget); } void ParametersToolBox::addParameter(QVBoxLayout * layout, const QString & key, const bool & value) { QCheckBox * widget = new QCheckBox(this); widget->setChecked(value); widget->setObjectName(key); connect(widget, SIGNAL(stateChanged(int)), this, SLOT(changeParameter(int))); addParameter(layout, key, widget); } void ParametersToolBox::addParameter(QVBoxLayout * layout, const QString & key, QWidget * widget) { QHBoxLayout * hLayout = new QHBoxLayout(); layout->insertLayout(layout->count()-1, hLayout); QString tmp = key.split('/').last(); if(tmp.at(0).isDigit()) { tmp.remove(0,1); } QLabel * label = new QLabel(tmp, this); label->setObjectName(key+"/label"); label->setToolTip(QString("%1").arg(Settings::getDescriptions().value(key, ""))); label->setTextInteractionFlags(Qt::TextSelectableByMouse); hLayout->addWidget(label); hLayout->addWidget(widget); } void ParametersToolBox::changeParameter(const QString & value) { if(sender()) { Settings::setParameter(sender()->objectName(), value); QStringList paramChanged; paramChanged.append(sender()->objectName()); Q_EMIT parametersChanged(paramChanged); } } void ParametersToolBox::changeParameter() { if(sender()) { QDoubleSpinBox * doubleSpinBox = qobject_cast(sender()); QSpinBox * spinBox = qobject_cast(sender()); QLineEdit * lineEdit = qobject_cast(sender()); if(doubleSpinBox) { Settings::setParameter(sender()->objectName(), doubleSpinBox->value()); } else if(spinBox) { if(spinBox->objectName().compare(Settings::kHomography_minimumInliers()) == 0 && spinBox->value() < 4) { Settings::setHomography_minimumInliers(4); spinBox->blockSignals(true); this->updateParameter(Settings::kHomography_minimumInliers()); spinBox->blockSignals(false); } else { Settings::setParameter(sender()->objectName(), spinBox->value()); } } else if(lineEdit) { Settings::setParameter(sender()->objectName(), lineEdit->text()); } QStringList paramChanged; paramChanged.append(sender()->objectName()); Q_EMIT parametersChanged(paramChanged); } } void ParametersToolBox::changeParameter(const int & value) { if(sender()) { QStringList paramChanged; QComboBox * comboBox = qobject_cast(sender()); QCheckBox * checkBox = qobject_cast(sender()); if(comboBox) { bool nnStrategyChanged = false; //verify binary issue with nearest neighbor strategy if(comboBox->objectName().compare(Settings::kFeature2D_2Descriptor()) == 0 || comboBox->objectName().compare(Settings::kNearestNeighbor_1Strategy()) == 0) { QComboBox * descriptorBox = (QComboBox*)this->getParameterWidget(Settings::kFeature2D_2Descriptor()); QComboBox * nnBox = (QComboBox*)this->getParameterWidget(Settings::kNearestNeighbor_1Strategy()); QComboBox * distBox = (QComboBox*)this->getParameterWidget(Settings::kNearestNeighbor_2Distance_type()); bool isBinaryDescriptor = descriptorBox->currentText().compare("ORB") == 0 || descriptorBox->currentText().compare("Brief") == 0 || descriptorBox->currentText().compare("BRISK") == 0 || descriptorBox->currentText().compare("FREAK") == 0; if(isBinaryDescriptor && nnBox->currentText().compare("Lsh") != 0) { QMessageBox::warning(this, tr("Warning"), tr("Current selected descriptor type (\"%1\") is binary while nearest neighbor strategy is not (\"%2\").\n" "Falling back to \"Lsh\" nearest neighbor strategy with Hamming distance (by default).") .arg(descriptorBox->currentText()) .arg(nnBox->currentText())); QString tmp = Settings::getNearestNeighbor_1Strategy(); *tmp.begin() = '5'; // set LSH Settings::setNearestNeighbor_1Strategy(tmp); tmp = Settings::getNearestNeighbor_2Distance_type(); *tmp.begin() = '8'; // set HAMMING Settings::setNearestNeighbor_2Distance_type(tmp); nnBox->blockSignals(true); distBox->blockSignals(true); this->updateParameter(Settings::kNearestNeighbor_1Strategy()); this->updateParameter(Settings::kNearestNeighbor_2Distance_type()); nnBox->blockSignals(false); distBox->blockSignals(false); if(sender() == nnBox) { this->updateParametersVisibility(); return; } nnStrategyChanged = true; paramChanged.append(Settings::kNearestNeighbor_1Strategy()); paramChanged.append(Settings::kNearestNeighbor_2Distance_type()); } else if(!isBinaryDescriptor && nnBox->currentText().compare("Lsh") == 0) { QMessageBox::warning(this, tr("Warning"), tr("Current selected descriptor type (\"%1\") is not binary while nearest neighbor strategy is (\"%2\").\n" "Falling back to \"KDTree\" nearest neighbor strategy with Euclidean_L2 distance (by default).") .arg(descriptorBox->currentText()) .arg(nnBox->currentText())); QString tmp = Settings::getNearestNeighbor_1Strategy(); *tmp.begin() = '1'; // set KDTree Settings::setNearestNeighbor_1Strategy(tmp); tmp = Settings::getNearestNeighbor_2Distance_type(); *tmp.begin() = '0'; // set EUCLIDEAN_L2 Settings::setNearestNeighbor_2Distance_type(tmp); nnBox->blockSignals(true); distBox->blockSignals(true); this->updateParameter(Settings::kNearestNeighbor_1Strategy()); this->updateParameter(Settings::kNearestNeighbor_2Distance_type()); nnBox->blockSignals(false); distBox->blockSignals(false); if(sender() == nnBox) { this->updateParametersVisibility(); return; } nnStrategyChanged = true; paramChanged.append(Settings::kNearestNeighbor_1Strategy()); paramChanged.append(Settings::kNearestNeighbor_2Distance_type()); } } // Distance issue when using nearest neighbor strategy using CV_32F type, though Lsh support all type (doesn't crash at least) if(nnStrategyChanged || comboBox->objectName().compare(Settings::kNearestNeighbor_1Strategy()) == 0 || comboBox->objectName().compare(Settings::kNearestNeighbor_2Distance_type()) == 0) { QComboBox * nnBox = (QComboBox*)this->getParameterWidget(Settings::kNearestNeighbor_1Strategy()); QComboBox * distBox = (QComboBox*)this->getParameterWidget(Settings::kNearestNeighbor_2Distance_type()); if(nnBox->currentText().compare("Lsh") != 0 && distBox->currentIndex() > 1) { QMessageBox::warning(this, tr("Warning"), tr("Current selected nearest neighbor strategy type (\"%1\") cannot handle distance strategy (\"%2\").\n" "Falling back to \"EUCLIDEAN_L2\" distance strategy (by default).") .arg(nnBox->currentText()) .arg(distBox->currentText())); QString tmp = Settings::getNearestNeighbor_2Distance_type(); *tmp.begin() = '0'; // set index Settings::setNearestNeighbor_2Distance_type(tmp); distBox->blockSignals(true); this->updateParameter(Settings::kNearestNeighbor_2Distance_type()); distBox->blockSignals(false); if(sender() == distBox) { this->updateParametersVisibility(); return; } paramChanged.append(Settings::kNearestNeighbor_2Distance_type()); } } QStringList items; for(int i=0; icount(); ++i) { items.append(comboBox->itemText(i)); } QString merged = QString::number(value) + QString(":") + items.join(";"); Settings::setParameter(sender()->objectName(), merged); this->updateParametersVisibility(); } else if(checkBox) { Settings::setParameter(sender()->objectName(), value==Qt::Checked?true:false); } paramChanged.append(sender()->objectName()); Q_EMIT parametersChanged(paramChanged); } } } // namespace find_object