Skip to content
MeasurementComplexItem.cc 25.7 KiB
Newer Older
Valentin Platzgummer's avatar
Valentin Platzgummer committed
#include "MeasurementComplexItem.h"

#include "CircularGenerator.h"
#include "LinearGenerator.h"
Valentin Platzgummer's avatar
Valentin Platzgummer committed
#include "RoutingThread.h"
#include "geometry/GenericCircle.h"
#include "geometry/MeasurementArea.h"
#include "geometry/SafeArea.h"
#include "geometry/clipper/clipper.hpp"
#include "geometry/snake.h"
#include "nemo_interface/SnakeTile.h"

Valentin Platzgummer's avatar
Valentin Platzgummer committed
// QGC
#include "JsonHelper.h"
Valentin Platzgummer's avatar
Valentin Platzgummer committed
#include "PlanMasterController.h"
Valentin Platzgummer's avatar
Valentin Platzgummer committed
#include "QGCApplication.h"
#include "QGCLoggingCategory.h"

// boost
#include <boost/units/io.hpp>
#include <boost/units/systems/si.hpp>

#define CLIPPER_SCALE 1000000
Valentin Platzgummer's avatar
Valentin Platzgummer committed
QGC_LOGGING_CATEGORY(MeasurementComplexItemLog, "MeasurementComplexItemLog")
Valentin Platzgummer's avatar
Valentin Platzgummer committed

template <typename T>
constexpr typename std::underlying_type<T>::type integral(T value) {
  return static_cast<typename std::underlying_type<T>::type>(value);
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
const char *MeasurementComplexItem::settingsGroup = "MeasurementComplexItem";
const char *MeasurementComplexItem::jsonComplexItemTypeValue =
    "MeasurementComplexItem";
const char *MeasurementComplexItem::variantName = "Variant";
const char *MeasurementComplexItem::altitudeName = "Altitude";
const QString MeasurementComplexItem::name(tr("Measurement"));

MeasurementComplexItem::MeasurementComplexItem(
    PlanMasterController *masterController, bool flyView,
    const QString &kmlOrShpFile, QObject *parent)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    : ComplexMissionItem(masterController, flyView, parent),
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      _masterController(masterController), _sequenceNumber(0),
      _followTerrain(false), _state(STATE::IDLE),
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      _metaDataMap(FactMetaData::createMapFromJsonFile(
Valentin Platzgummer's avatar
Valentin Platzgummer committed
          QStringLiteral(":/json/MeasurementComplexItem.SettingsGroup.json"),
          this)),
      _altitude(settingsGroup, _metaDataMap[altitudeName]),
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      _variant(settingsGroup, _metaDataMap[variantName]),
      _pAreaData(new AreaData(this)), _pEditorData(new AreaData(this)),
      _pCurrentData(_pAreaData), _pGenerator(nullptr),
      _pWorker(new RoutingThread(this)) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed

  Q_UNUSED(kmlOrShpFile)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  _editorQml = "qrc:/qml/MeasurementItemEditor.qml";
Valentin Platzgummer's avatar
Valentin Platzgummer committed

  // Connect facts.
  connect(&this->_variant, &Fact::rawValueChanged, this,
Valentin Platzgummer's avatar
Valentin Platzgummer committed
          &MeasurementComplexItem::_changeVariant);
Valentin Platzgummer's avatar
Valentin Platzgummer committed

  // Connect worker.
  connect(this->_pWorker, &RoutingThread::result, this,
Valentin Platzgummer's avatar
Valentin Platzgummer committed
          &MeasurementComplexItem::_storeRoutingData);

  // Connect coordinate and exitCoordinate.
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  connect(this, &MeasurementComplexItem::routeChanged,
          [this] { emit this->coordinateChanged(this->coordinate()); });
  connect(this, &MeasurementComplexItem::routeChanged,
          [this] { emit this->exitCoordinateChanged(this->exitCoordinate()); });
  connect(this, &MeasurementComplexItem::routeChanged, [this] {
    emit this->exitCoordinateSameAsEntryChanged(
        this->exitCoordinateSameAsEntry());
  });
  // Connect complexDistance.
  connect(this, &MeasurementComplexItem::routeChanged,
          [this] { emit this->complexDistanceChanged(); });

Valentin Platzgummer's avatar
Valentin Platzgummer committed
  // Register Generators.
  auto lg = new routing::LinearGenerator(this->_pAreaData, this);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  registerGenerator(lg->name(), lg);
  auto cg = new routing::CircularGenerator(this->_pAreaData, this);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  registerGenerator(cg->name(), cg);
Valentin Platzgummer's avatar
Valentin Platzgummer committed

  qCritical() << "ToDo: _altitude connections missing.";
  qCritical() << "ToDo: add generator saveing.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
MeasurementComplexItem::~MeasurementComplexItem() {}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::reverseRoute() { _reverseRoute(); }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
const AreaData *MeasurementComplexItem::areaData() const {
  return this->_pCurrentData;
AreaData *MeasurementComplexItem::areaData() { return this->_pCurrentData; }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
QVariantList MeasurementComplexItem::route() { return _route; }

Valentin Platzgummer's avatar
Valentin Platzgummer committed
QStringList MeasurementComplexItem::variantNames() const {
  return _variantNames;
}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::load(const QJsonObject &complexObject,
                                  int sequenceNumber, QString &errorString) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  qWarning() << "MeasurementComplexItem::load(): area data load missing.";
  qWarning() << "MeasurementComplexItem::load(): mission item load missing.";
  qWarning() << "MeasurementComplexItem::load(): add editingStart/Stop.";

Valentin Platzgummer's avatar
Valentin Platzgummer committed
  // We need to pull version first to determine what validation/conversion
  // needs to be performed
  QList<JsonHelper::KeyValidateInfo> versionKeyInfoList = {
      {JsonHelper::jsonVersionKey, QJsonValue::Double, true},
  };
  if (!JsonHelper::validateKeys(complexObject, versionKeyInfoList,
                                errorString)) {
    return false;
  }

  int version = complexObject[JsonHelper::jsonVersionKey].toInt();
  if (version != 1) {
    errorString = tr("Survey items do not support version %1").arg(version);
    return false;
  }

  QList<JsonHelper::KeyValidateInfo> keyInfoList = {
      {VisualMissionItem::jsonTypeKey, QJsonValue::String, true},
      {ComplexMissionItem::jsonComplexItemTypeKey, QJsonValue::String, true},
      {variantName, QJsonValue::Double, false},
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      {altitudeName, QJsonValue::Double, true},
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  };

  if (!JsonHelper::validateKeys(complexObject, keyInfoList, errorString)) {
    return false;
  }

  QString itemType = complexObject[VisualMissionItem::jsonTypeKey].toString();
  QString complexType =
      complexObject[ComplexMissionItem::jsonComplexItemTypeKey].toString();
  if (itemType != VisualMissionItem::jsonTypeComplexItemValue ||
      complexType != jsonComplexItemTypeValue) {
    errorString = tr("%1 does not support loading this complex mission item "
                     "type: %2:%3")
                      .arg(qgcApp()->applicationName())
                      .arg(itemType)
                      .arg(complexType);
    return false;
  }

  setSequenceNumber(sequenceNumber);

  if (!load(complexObject, sequenceNumber, errorString)) {
    return false;
  }

  _variant.setRawValue(complexObject[variantName].toInt());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  _altitude.setRawValue(complexObject[altitudeName].toDouble());
double
MeasurementComplexItem::greatestDistanceTo(const QGeoCoordinate &other) const {
  double d = -1 * std::numeric_limits<double>::infinity();
  if (other.isValid()) {
    if (this->_route.size() > 0) {
      std::for_each(this->_route.cbegin(), this->_route.cend(),
                    [&d, &other](const QVariant &variant) {
                      auto vertex = variant.value<QGeoCoordinate>();
                      d = std::max(d, vertex.distanceTo(other));
                    });
    }
  } else {
    qCDebug(MeasurementComplexItemLog)
        << "greatestDistanceTo(): invalid QGeoCoordinate: " << other;
  }
  return d;
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::dirty() const { return _dirty; }

bool MeasurementComplexItem::isSimpleItem() const { return false; }

bool MeasurementComplexItem::isStandaloneCoordinate() const { return false; }

QString MeasurementComplexItem::mapVisualQML() const {
  return QStringLiteral("MeasurementItemMapVisual.qml");
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::save(QJsonArray &planItems) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  qWarning() << "MeasurementComplexItem::save(): area data save missing.";
  qWarning() << "MeasurementComplexItem::save(): mission item save missing.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (ready()) {
    QJsonObject saveObject;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    saveObject[JsonHelper::jsonVersionKey] = 1;
    saveObject[VisualMissionItem::jsonTypeKey] =
        VisualMissionItem::jsonTypeComplexItemValue;
    saveObject[ComplexMissionItem::jsonComplexItemTypeKey] =
        jsonComplexItemTypeValue;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    saveObject[variantName] = double(_variant.rawValue().toUInt());
    saveObject[altitudeName] = double(_altitude.rawValue().toUInt());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    planItems.append(saveObject);
  }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
double MeasurementComplexItem::amslEntryAlt() const {
  return _altitude.rawValue().toDouble() +
         this->_masterController->missionController()
             ->plannedHomePosition()
             .altitude();
}

double MeasurementComplexItem::amslExitAlt() const { return amslEntryAlt(); }

double MeasurementComplexItem::minAMSLAltitude() const {
  return amslEntryAlt();
}

double MeasurementComplexItem::maxAMSLAltitude() const {
  return amslEntryAlt();
}

QString MeasurementComplexItem::commandDescription() const {
  return QStringLiteral("Measurement");
}

QString MeasurementComplexItem::commandName() const {
  return QStringLiteral("Measurement");
}

QString MeasurementComplexItem::abbreviation() const {
  return QStringLiteral("M");
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::specifiesCoordinate() const {
  return _route.count() > 0;
}

bool MeasurementComplexItem::specifiesAltitudeOnly() const { return false; }

QGeoCoordinate MeasurementComplexItem::coordinate() const {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  return this->_route.size() > 0 ? _route.first().value<QGeoCoordinate>()
                                 : QGeoCoordinate();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
}

QGeoCoordinate MeasurementComplexItem::exitCoordinate() const {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  return this->_route.size() > 0 ? _route.last().value<QGeoCoordinate>()
                                 : QGeoCoordinate();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
}

int MeasurementComplexItem::sequenceNumber() const { return _sequenceNumber; }

double MeasurementComplexItem::specifiedFlightSpeed() {
  return std::numeric_limits<double>::quiet_NaN();
}

double MeasurementComplexItem::specifiedGimbalYaw() {
  return std::numeric_limits<double>::quiet_NaN();
}

double MeasurementComplexItem::specifiedGimbalPitch() {
  return std::numeric_limits<double>::quiet_NaN();
}

void MeasurementComplexItem::appendMissionItems(QList<MissionItem *> &items,
                                                QObject *missionItemParent) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed

  if (ready()) {
    qCDebug(MeasurementComplexItemLog) << "appendMissionItems()";

    int seqNum = this->_sequenceNumber;

    MAV_FRAME mavFrame =
        followTerrain() ? MAV_FRAME_GLOBAL : MAV_FRAME_GLOBAL_RELATIVE_ALT;

    for (const auto &variant : this->_route) {
      auto vertex = variant.value<QGeoCoordinate>();
      MissionItem *item = new MissionItem(
          seqNum++, MAV_CMD_NAV_WAYPOINT, mavFrame,
          0,   // hold time
          0.0, // No acceptance radius specified
          0.0, // Pass through waypoint
          std::numeric_limits<double>::quiet_NaN(), // Yaw unchanged
          vertex.latitude(), vertex.longitude(), vertex.altitude(),
          true,  // autoContinue
          false, // isCurrentItem
          missionItemParent);
      items.append(item);
    }
  } else {
    qCDebug(MeasurementComplexItemLog)
        << "appendMissionItems(): called while not ready().";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  }
}

void MeasurementComplexItem::setMissionFlightStatus(
    const MissionController::MissionFlightStatus_t &missionFlightStatus) {
  ComplexMissionItem::setMissionFlightStatus(missionFlightStatus);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::applyNewAltitude(double newAltitude) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  this->_altitude.setRawValue(newAltitude);
  qWarning() << "applyNewAltitude(): impl. missing.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
}

double MeasurementComplexItem::additionalTimeDelay() const { return 0; }

bool MeasurementComplexItem::_setGenerator(PtrGenerator newG) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (this->_pGenerator != newG) {
    if (this->_pGenerator != nullptr) {
      disconnect(this->_pGenerator, &routing::GeneratorBase::generatorChanged,
Valentin Platzgummer's avatar
Valentin Platzgummer committed
                 this, &MeasurementComplexItem::_updateRoute);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    }

    this->_pGenerator = newG;

    if (this->_pGenerator != nullptr) {
      connect(this->_pGenerator, &routing::GeneratorBase::generatorChanged,
              this, &MeasurementComplexItem::_updateRoute);
    }

Valentin Platzgummer's avatar
Valentin Platzgummer committed
    emit generatorChanged();

Valentin Platzgummer's avatar
Valentin Platzgummer committed
    if (!editing()) {
      this->_setState(STATE::IDLE);
      _updateRoute();
    }
Valentin Platzgummer's avatar
Valentin Platzgummer committed

    return true;
  } else {
    return false;
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::_setState(MeasurementComplexItem::STATE state) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (this->_state != state) {
    auto oldState = this->_state;
    this->_state = state;
Valentin Platzgummer's avatar
Valentin Platzgummer committed

Valentin Platzgummer's avatar
Valentin Platzgummer committed
    if (_calculating(oldState) != _calculating(state)) {
      emit calculatingChanged();
    }
Valentin Platzgummer's avatar
Valentin Platzgummer committed

    if (_editing(oldState) != _editing(state)) {
      emit editingChanged();
    }
Valentin Platzgummer's avatar
Valentin Platzgummer committed

    if (_ready(oldState) != _ready(state)) {
      emit readyChanged();
    }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::_calculating(MeasurementComplexItem::STATE state) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  return state == STATE::ROUTING;
}

bool MeasurementComplexItem::_editing(MeasurementComplexItem::STATE state) {
  return state == STATE::EDITING;
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::_ready(MeasurementComplexItem::STATE state) {
  return state == STATE::IDLE;
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::_setAreaData(
    MeasurementComplexItem::PtrAreaData data) {
  if (_pCurrentData != data) {
    _pCurrentData = data;
    emit areaDataChanged();
  }
}

void MeasurementComplexItem::_updateRoute() {
  if (!editing()) {
    // Reset data.
    this->_route.clear();
    this->_variantVector.clear();
    this->_variantNames.clear();
    emit variantNamesChanged();

    if (this->_pAreaData->isCorrect()) {

      // Prepare data.
      auto origin = this->_pAreaData->origin();
      origin.setAltitude(0);
      if (!origin.isValid()) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        qCDebug(MeasurementComplexItemLog)
            << "_updateWorker(): origin invalid." << origin;
        return;
      // Convert safe area.
      auto serviceArea =
          getGeoArea<const SafeArea *>(*this->_pAreaData->areaList());
      auto geoSafeArea = serviceArea->coordinateList();
      if (!(geoSafeArea.size() >= 3)) {
        qCDebug(MeasurementComplexItemLog)
            << "_updateWorker(): safe area invalid." << geoSafeArea;
        return;
      }
      for (auto &v : geoSafeArea) {
        if (v.isValid()) {
          v.setAltitude(0);
        } else {
          qCDebug(MeasurementComplexItemLog)
              << "_updateWorker(): safe area contains invalid coordinate."
              << geoSafeArea;
          return;
        }
      }
      // Routing par.
      RoutingParameter par;
      par.numSolutions = 5;
      auto &safeAreaENU = par.safeArea;
      snake::areaToEnu(origin, geoSafeArea, safeAreaENU);

      // Create generator.
      if (this->_pGenerator != nullptr) {
        routing::GeneratorBase::Generator g; // Transect generator.
        if (this->_pGenerator->get(g)) {
          // Start/Restart routing worker.
          this->_pWorker->route(par, g);
          _setState(STATE::ROUTING);
          return;
        } else {
          qCDebug(MeasurementComplexItemLog)
              << "_updateWorker(): generator creation failed.";
          return;
        }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        qCDebug(MeasurementComplexItemLog)
            << "_updateWorker(): pGenerator == nullptr, number of registered "
               "generators: "
            << this->_generatorList.size();
        return;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      }
    } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      qCDebug(MeasurementComplexItemLog)
          << "_updateWorker(): plan data invalid.";
      return;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::_changeVariant() {
  if (ready()) {
    auto variant = this->_variant.rawValue().toUInt();

    // Find old variant and run. Old run corresponts with empty list.
    std::size_t old_variant = std::numeric_limits<std::size_t>::max();
    for (std::size_t i = 0; i < std::size_t(this->_variantVector.size()); ++i) {
      const auto &variantCoordinates = this->_variantVector.at(i);
      if (variantCoordinates.isEmpty()) {
        old_variant = i;
        break;
      }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    // Swap route.
    if (variant != old_variant) {
      // Swap in new variant.
      if (variant < std::size_t(this->_variantVector.size())) {
        if (old_variant != std::numeric_limits<std::size_t>::max()) {
          // this->_route containes a route, swap it back to
          // this->_solutionVector
          auto &oldVariantCoordinates = this->_variantVector[old_variant];
          oldVariantCoordinates.swap(this->_route);
        }
        auto &newVariantCoordinates = this->_variantVector[variant];
        this->_route.swap(newVariantCoordinates);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      } else { // error
        qCDebug(MeasurementComplexItemLog)
            << "Variant out of bounds (variant =" << variant << ").";
        qCDebug(MeasurementComplexItemLog) << "Resetting variant to zero.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        disconnect(&this->_variant, &Fact::rawValueChanged, this,
                   &MeasurementComplexItem::_changeVariant);
        this->_variant.setCookedValue(QVariant(0));
        connect(&this->_variant, &Fact::rawValueChanged, this,
                &MeasurementComplexItem::_changeVariant);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        if (this->_variantVector.size() > 0) {
          this->_changeVariant();
        }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::_reverseRoute() {
  if (ready()) {
    if (this->_route.size() > 0) {
      auto &t = this->_route;
      std::reverse(t.begin(), t.end());
    }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
ComplexMissionItem::ReadyForSaveState
Valentin Platzgummer's avatar
Valentin Platzgummer committed
MeasurementComplexItem::readyForSaveState() const {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (ready()) {
    return ReadyForSaveState::ReadyForSave;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return ReadyForSaveState::NotReadyForSaveData;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::exitCoordinateSameAsEntry() const {
  return this->_route.size() > 0 ? this->_route.first() == this->_route.last()
                                 : false;
}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::setDirty(bool dirty) {
  if (this->_dirty != dirty) {
    this->_dirty = dirty;
    emit dirtyChanged(this->_dirty);
  }
}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::setCoordinate(const QGeoCoordinate &coordinate) {
  Q_UNUSED(coordinate);
}

void MeasurementComplexItem::setSequenceNumber(int sequenceNumber) {
  if (this->_sequenceNumber != sequenceNumber) {
    this->_sequenceNumber = sequenceNumber;
    emit sequenceNumberChanged(this->_sequenceNumber);
  }
}

QString MeasurementComplexItem::patternName() const { return name; }

double MeasurementComplexItem::complexDistance() const {
  double d = 0;
  if (this->_route.size() > 1) {
    auto vertex = _route.first().value<QGeoCoordinate>();
    std::for_each(this->_route.cbegin() + 1, this->_route.cend(),
                  [&vertex, &d](const QVariant &variant) {
                    auto otherVertex = variant.value<QGeoCoordinate>();
                    d += vertex.distanceTo(otherVertex);
                    vertex = otherVertex;
                  });
  }
  return d;
}

int MeasurementComplexItem::lastSequenceNumber() const {
  return _sequenceNumber + std::max(0, this->_route.size() - 1);
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::registerGenerator(const QString &name,
                                               routing::GeneratorBase *g) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (name.isEmpty()) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog)
        << "registerGenerator(): empty name string.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return false;
  }

  if (!g) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog)
        << "registerGenerator(): empty generator.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return false;
  }

  if (this->_generatorNameList.contains(name)) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog) << "registerGenerator(): generator "
                                          "already registered.";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return false;
  } else {
    this->_generatorNameList.push_back(name);
    this->_generatorList.push_back(g);
    if (this->_generatorList.size() == 1) {
      _setGenerator(g);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    }

    emit generatorNameListChanged();
    return true;
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::unregisterGenerator(const QString &name) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  auto index = this->_generatorNameList.indexOf(name);
  if (index >= 0) {
    // Is this the current generator?
    const auto &g = this->_generatorList.at(index);
    if (g == this->_pGenerator) {
      if (index > 0) {
        _setGenerator(this->_generatorList.at(index - 1));
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      } else {
        _setGenerator(nullptr);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
            << "unregisterGenerator(): last generator unregistered.";
      }
    }

    this->_generatorNameList.removeAt(index);
    auto gen = this->_generatorList.takeAt(index);
    gen->deleteLater();
Valentin Platzgummer's avatar
Valentin Platzgummer committed

    emit generatorNameListChanged();
    return true;
  } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        << "unregisterGenerator(): generator " << name << " not registered.";
    return false;
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::unregisterGenerator(int index) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (index > 0 && index < this->_generatorNameList.size()) {
    return unregisterGenerator(this->_generatorNameList.at(index));
  } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog)
        << "unregisterGenerator(): index (" << index
        << ") out"
           "of bounds ( "
        << this->_generatorList.size() << " ).";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return false;
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::switchToGenerator(const QString &name) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  auto index = this->_generatorNameList.indexOf(name);
  if (index >= 0) {
    _setGenerator(this->_generatorList.at(index));
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return true;
  } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        << "switchToGenerator(): generator " << name << " not registered.";
    return false;
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::switchToGenerator(int index) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (index >= 0) {
    _setGenerator(this->_generatorList.at(index));
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return true;
  } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        << "switchToGenerator(): index (" << index
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        << ") out"
           "of bounds ( "
        << this->_generatorNameList.size() << " ).";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    return false;
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
QStringList MeasurementComplexItem::generatorNameList() {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  return this->_generatorNameList;
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
routing::GeneratorBase *MeasurementComplexItem::generator() {
  return _pGenerator;
}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
int MeasurementComplexItem::generatorIndex() {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  return this->_generatorList.indexOf(this->_pGenerator);
}

void MeasurementComplexItem::startEditing() {
  if (!editing()) {
    *_pEditorData = *_pAreaData;
    _setAreaData(_pEditorData);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    _setState(STATE::EDITING);
void MeasurementComplexItem::stopEditing() {
  if (editing()) {
    bool correct = _pEditorData->isCorrect();
    if (correct) {
      *_pAreaData = *_pEditorData;
    _setAreaData(_pAreaData);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    _setState(STATE::IDLE);
    if (correct && *_pEditorData != *_pAreaData) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      _updateRoute();
void MeasurementComplexItem::abortEditing() {
  if (editing()) {
    _setAreaData(_pAreaData);
    _setState(STATE::IDLE);
  }
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
void MeasurementComplexItem::_storeRoutingData(
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    MeasurementComplexItem::PtrRoutingData pRoute) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  if (this->_state == STATE::ROUTING) {
    // Store solutions.
    auto ori = this->_pAreaData->origin();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    ori.setAltitude(0);
    const auto &transectsENU = pRoute->transects;
    QVector<Variant> variantVector;
    const auto nSolutions = pRoute->solutionVector.size();

    for (std::size_t j = 0; j < nSolutions; ++j) {
      Variant var;
      const auto &solution = pRoute->solutionVector.at(j);
      if (solution.size() > 0) {
        const auto &route = solution.at(0);
        const auto &path = route.path;
        const auto &info = route.info;
        if (info.size() > 1) {
          // Find index of first waypoint.
          std::size_t idxFirst = 0;
          const auto &infoFirst = info.at(1);
          const auto &firstTransect = transectsENU[infoFirst.index];
          if (firstTransect.size() > 0) {
            const auto &firstWaypoint = infoFirst.reversed
                                            ? firstTransect.back()
                                            : firstTransect.front();
            double th = 0.01;
            for (std::size_t i = 0; i < path.size(); ++i) {
              auto dist = bg::distance(path[i], firstWaypoint);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
              if (dist < th) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
                idxFirst = i;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
            // Find index of last waypoint.
            std::size_t idxLast = path.size() - 1;
            const auto &infoLast = info.at(info.size() - 2);
            const auto &lastTransect = transectsENU[infoLast.index];
            if (lastTransect.size() > 0) {
              const auto &lastWaypoint = infoLast.reversed
                                             ? lastTransect.front()
                                             : lastTransect.back();
              for (long i = path.size() - 1; i >= 0; --i) {
                auto dist = bg::distance(path[i], lastWaypoint);
                if (dist < th) {
                  idxLast = i;
                  break;
                }
              }

              // Convert to geo coordinates.
              for (std::size_t i = idxFirst; i <= idxLast; ++i) {
                auto &vertex = path[i];
                QGeoCoordinate c;
                snake::fromENU(ori, vertex, c);
                var.append(QVariant::fromValue(c));
              }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
            } else {
              qCDebug(MeasurementComplexItemLog)
                  << "_setTransects(): lastTransect.size() == 0";
            }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
          } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
            qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
                << "_setTransects(): firstTransect.size() == 0";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
          }
        } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
          qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
              << "_setTransects(): transectsInfo.size() <= 1";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        }
      } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
        qCDebug(MeasurementComplexItemLog)
Valentin Platzgummer's avatar
Valentin Platzgummer committed
            << "_setTransects(): solution.size() == 0";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      if (var.size() > 0) {
        variantVector.push_back(std::move(var));
      }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    // Assign routes if no error occured.
    if (variantVector.size() > 0) {
      // Swap first route to _route.
      this->_variantVector.swap(variantVector);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      // Add route variant names.
      this->_variantNames.clear();
      for (std::size_t i = 1; i <= std::size_t(this->_variantVector.size());
           ++i) {
        this->_variantNames.append(QString::number(i));
      }
      emit variantNamesChanged();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      disconnect(&this->_variant, &Fact::rawValueChanged, this,
                 &MeasurementComplexItem::_changeVariant);
      this->_variant.setCookedValue(QVariant(0));
      connect(&this->_variant, &Fact::rawValueChanged, this,
              &MeasurementComplexItem::_changeVariant);
      this->_route.swap(this->_variantVector.first());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
      emit routeChanged();

      this->_setState(STATE::IDLE);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
    } else {
      qCDebug(MeasurementComplexItemLog)
          << "_setTransects(): failed, variantVector empty.";
      this->_setState(STATE::IDLE);
    }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
Fact *MeasurementComplexItem::variant() { return &_variant; }

Fact *MeasurementComplexItem::altitude() { return &this->_altitude; }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::calculating() const {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
  return this->_calculating(this->_state);
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::editing() const { return _editing(this->_state); }

bool MeasurementComplexItem::ready() const { return _ready(this->_state); }

Valentin Platzgummer's avatar
Valentin Platzgummer committed
bool MeasurementComplexItem::followTerrain() const { return _followTerrain; }