WimaMeasurementArea.cc 12.4 KB
Newer Older
1
#include "WimaMeasurementArea.h"
2
#include "QtConcurrentRun"
Valentin Platzgummer's avatar
Valentin Platzgummer committed
3 4 5 6
#include "SnakeTile.h"
#include "snake.h"

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

8 9 10 11
#ifndef SNAKE_MAX_TILES
#define SNAKE_MAX_TILES 1000
#endif

12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
TileData::TileData() : tiles(this) {}

TileData::~TileData() { tiles.clearAndDeleteContents(); }

TileData &TileData::operator=(const TileData &other) {
  this->tiles.clearAndDeleteContents();
  for (std::size_t i = 0; i < std::size_t(other.tiles.count()); ++i) {
    const auto *obj = other.tiles.get(i);
    const auto *tile = qobject_cast<const SnakeTile *>(obj);
    if (tile != nullptr) {
      this->tiles.append(new SnakeTile(*tile, this));
    } else {
      qWarning("TileData::operator=: nullptr");
    }
  }
  this->tileCenterPoints = other.tileCenterPoints;
  return *this;
}

void TileData::clear() {
  this->tiles.clearAndDeleteContents();
  this->tileCenterPoints.clear();
}

size_t TileData::size() const {
  if (tiles.count() == tileCenterPoints.size()) {
    return tiles.count();
  } else {
    return 0;
  }
}

44
const char *WimaMeasurementArea::settingsGroup = "MeasurementArea";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
45 46 47 48 49 50
const char *WimaMeasurementArea::tileHeightName = "TileHeight";
const char *WimaMeasurementArea::tileWidthName = "TileWidth";
const char *WimaMeasurementArea::minTileAreaName = "MinTileArea";
const char *WimaMeasurementArea::transectDistanceName = "TransectDistance";
const char *WimaMeasurementArea::minTransectLengthName = "MinTransectLength";
const char *WimaMeasurementArea::showTilesName = "ShowTiles";
51
const char *WimaMeasurementArea::WimaMeasurementAreaName = "Measurement Area";
Valentin Platzgummer's avatar
Valentin Platzgummer committed
52

53 54
void tileDeleter(QmlObjectListModel *tiles) { tiles->clearAndDeleteContents(); }

Valentin Platzgummer's avatar
Valentin Platzgummer committed
55
WimaMeasurementArea::WimaMeasurementArea(QObject *parent)
56 57 58 59
    : WimaArea(parent),
      _metaDataMap(FactMetaData::createMapFromJsonFile(
          QStringLiteral(":/json/WimaMeasurementArea.SettingsGroup.json"),
          this /* QObject parent */)),
Valentin Platzgummer's avatar
Valentin Platzgummer committed
60 61 62 63 64 65 66 67 68 69 70 71 72 73
      _tileHeight(SettingsFact(settingsGroup, _metaDataMap[tileHeightName],
                               this /* QObject parent */)),
      _tileWidth(SettingsFact(settingsGroup, _metaDataMap[tileWidthName],
                              this /* QObject parent */)),
      _minTileArea(SettingsFact(settingsGroup, _metaDataMap[minTileAreaName],
                                this /* QObject parent */)),
      _transectDistance(SettingsFact(settingsGroup,
                                     _metaDataMap[transectDistanceName],
                                     this /* QObject parent */)),
      _minTransectLength(SettingsFact(settingsGroup,
                                      _metaDataMap[minTransectLengthName],
                                      this /* QObject parent */)),
      _showTiles(SettingsFact(settingsGroup, _metaDataMap[showTilesName],
                              this /* QObject parent */)),
74
      _calculating(false) {
75
  init();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
76 77
}

78 79 80 81 82 83
WimaMeasurementArea::WimaMeasurementArea(const WimaMeasurementArea &other,
                                         QObject *parent)
    : WimaArea(other, parent),
      _metaDataMap(FactMetaData::createMapFromJsonFile(
          QStringLiteral(":/json/WimaMeasurementArea.SettingsGroup.json"),
          this /* QObject parent */)),
Valentin Platzgummer's avatar
Valentin Platzgummer committed
84 85 86 87 88 89 90 91 92 93 94 95 96 97
      _tileHeight(SettingsFact(settingsGroup, _metaDataMap[tileHeightName],
                               this /* QObject parent */)),
      _tileWidth(SettingsFact(settingsGroup, _metaDataMap[tileWidthName],
                              this /* QObject parent */)),
      _minTileArea(SettingsFact(settingsGroup, _metaDataMap[minTileAreaName],
                                this /* QObject parent */)),
      _transectDistance(SettingsFact(settingsGroup,
                                     _metaDataMap[transectDistanceName],
                                     this /* QObject parent */)),
      _minTransectLength(SettingsFact(settingsGroup,
                                      _metaDataMap[minTransectLengthName],
                                      this /* QObject parent */)),
      _showTiles(SettingsFact(settingsGroup, _metaDataMap[showTilesName],
                              this /* QObject parent */)),
98
      _calculating(false) {
99
  init();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
100 101
}

102 103 104 105 106
/*!
 * \overload operator=()
 *
 * Calls the inherited operator WimaArea::operator=().
 */
107 108 109 110 111 112 113
WimaMeasurementArea &WimaMeasurementArea::
operator=(const WimaMeasurementArea &other) {
  WimaArea::operator=(other);

  return *this;
}

114
WimaMeasurementArea::~WimaMeasurementArea() {}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
115

116 117 118
QString WimaMeasurementArea::mapVisualQML() const {
  return "WimaMeasurementAreaMapVisual.qml";
}
119

120 121
QString WimaMeasurementArea::editorQML() const {
  return "WimaMeasurementAreaEditor.qml";
122 123
}

124 125
Fact *WimaMeasurementArea::tileHeight() { return &_tileHeight; }

Valentin Platzgummer's avatar
Valentin Platzgummer committed
126 127 128 129 130 131 132 133 134 135
Fact *WimaMeasurementArea::tileWidth() { return &_tileWidth; }

Fact *WimaMeasurementArea::minTileArea() { return &_minTileArea; }

Fact *WimaMeasurementArea::transectDistance() { return &_transectDistance; }

Fact *WimaMeasurementArea::minTransectLength() { return &_minTransectLength; }

Fact *WimaMeasurementArea::showTiles() { return &_showTiles; }

136 137 138
QmlObjectListModel *WimaMeasurementArea::tiles() {
  return &this->_tileData.tiles;
}
139

140
const QmlObjectListModel *WimaMeasurementArea::tiles() const {
141 142 143 144 145 146 147 148 149
  return &this->_tileData.tiles;
}

const QVariantList &WimaMeasurementArea::tileCenterPoints() const {
  return this->_tileData.tileCenterPoints;
}

const TileData &WimaMeasurementArea::tileData() const {
  return this->_tileData;
150 151 152
}

int WimaMeasurementArea::maxTiles() const { return SNAKE_MAX_TILES; }
153

154
bool WimaMeasurementArea::ready() const { return !_calculating; }
155

156 157
void WimaMeasurementArea::saveToJson(QJsonObject &json) {
  this->WimaArea::saveToJson(json);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
158 159 160 161 162 163
  json[tileHeightName] = _tileHeight.rawValue().toDouble();
  json[tileWidthName] = _tileWidth.rawValue().toDouble();
  json[minTileAreaName] = _minTileArea.rawValue().toDouble();
  json[transectDistanceName] = _transectDistance.rawValue().toDouble();
  json[minTransectLengthName] = _minTransectLength.rawValue().toDouble();
  json[showTilesName] = _showTiles.rawValue().toBool();
164
  json[areaTypeName] = WimaMeasurementAreaName;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
165 166
}

167 168 169 170 171
bool WimaMeasurementArea::loadFromJson(const QJsonObject &json,
                                       QString &errorString) {
  if (this->WimaArea::loadFromJson(json, errorString)) {
    bool retVal = true;

Valentin Platzgummer's avatar
Valentin Platzgummer committed
172 173
    if (json.contains(tileHeightName) && json[tileHeightName].isDouble()) {
      _tileHeight.setRawValue(json[tileHeightName].toDouble());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
174
    } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
175
      errorString.append(tr("Could not load tile height!\n"));
176
      retVal = false;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
177 178
    }

Valentin Platzgummer's avatar
Valentin Platzgummer committed
179
    if (json.contains(tileWidthName) && json[tileWidthName].isDouble()) {
180
      _tileWidth.setRawValue(json[tileWidthName].toDouble());
181
    } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
182
      errorString.append(tr("Could not load tile width!\n"));
183 184
      retVal = false;
    }
185

Valentin Platzgummer's avatar
Valentin Platzgummer committed
186
    if (json.contains(minTileAreaName) && json[minTileAreaName].isDouble()) {
187
      _minTileArea.setRawValue(json[minTileAreaName].toDouble());
188
    } else {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
189
      errorString.append(tr("Could not load minimal tile area!\n"));
190
      retVal = false;
191
    }
192

Valentin Platzgummer's avatar
Valentin Platzgummer committed
193 194
    if (json.contains(transectDistanceName) &&
        json[transectDistanceName].isDouble()) {
195
      _transectDistance.setRawValue(json[transectDistanceName].toDouble());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
196 197 198 199 200 201 202
    } else {
      errorString.append(tr("Could not load transect distance!\n"));
      retVal = false;
    }

    if (json.contains(minTransectLengthName) &&
        json[minTransectLengthName].isDouble()) {
203
      _minTransectLength.setRawValue(json[minTransectLengthName].toDouble());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
204 205 206 207 208 209
    } else {
      errorString.append(tr("Could not load minimal transect length!\n"));
      retVal = false;
    }

    if (json.contains(showTilesName) && json[showTilesName].isBool()) {
210
      _showTiles.setRawValue(json[showTilesName].toBool());
Valentin Platzgummer's avatar
Valentin Platzgummer committed
211 212 213 214
    } else {
      errorString.append(tr("Could not load show tiles !\n"));
      retVal = false;
    }
215 216 217 218
    return retVal;
  } else {
    return false;
  }
219
}
220 221
//!
//! \brief WimaMeasurementArea::doUpdate
222 223
//! \pre WimaMeasurementArea::deferUpdate must be called first, don't call
//! this function directly!
Valentin Platzgummer's avatar
Valentin Platzgummer committed
224 225 226 227 228 229
void WimaMeasurementArea::doUpdate() {
  using namespace snake;
  using namespace boost::units;
#ifdef SNAKE_SHOW_TIME
  auto start = std::chrono::high_resolution_clock::now();
#endif
230 231 232 233 234 235 236 237 238
  const auto height = this->_tileHeight.rawValue().toDouble() * si::meter;
  const auto width = this->_tileWidth.rawValue().toDouble() * si::meter;
  const auto tileArea = width * height;
  const auto totalArea = this->area() * si::meter * si::meter;
  const auto estNumTiles = totalArea / tileArea;
  if (!this->_calculating &&
      long(std::ceil(estNumTiles.value())) <= SNAKE_MAX_TILES &&
      this->count() >= 3 && this->isSimplePolygon()) {
    this->_calculating = true;
239 240 241
    auto polygon = this->coordinateList();
    for (auto &v : polygon) {
      v.setAltitude(0);
242 243
    }
    const auto minArea =
Valentin Platzgummer's avatar
Valentin Platzgummer committed
244
        this->_minTileArea.rawValue().toDouble() * si::meter * si::meter;
245 246 247 248 249
    auto *th = this->thread();
    auto future = QtConcurrent::run([polygon, th, height, width, minArea] {
#ifdef SNAKE_SHOW_TIME
      auto start = std::chrono::high_resolution_clock::now();
#endif
250 251
      DataPtr pData(new TileData());
      // Convert to ENU system.
252 253 254 255 256 257
      QGeoCoordinate origin = polygon.first();
      BoostPolygon polygonENU;
      areaToEnu(origin, polygon, polygonENU);
      std::vector<BoostPolygon> tilesENU;
      BoundingBox bbox;
      std::string errorString;
258
      // Generate tiles.
259 260
      if (snake::tiles(polygonENU, height, width, minArea, tilesENU, bbox,
                       errorString)) {
261
        // Convert to geo system.
262
        for (const auto &t : tilesENU) {
263
          auto geoTile = new SnakeTile(pData.get());
264 265 266 267 268
          for (const auto &v : t.outer()) {
            QGeoCoordinate geoVertex;
            fromENU(origin, v, geoVertex);
            geoTile->push_back(geoVertex);
          }
269 270 271 272 273 274 275
          pData->tiles.append(geoTile);
          // Calculate center.
          snake::BoostPoint center;
          snake::polygonCenter(t, center);
          QGeoCoordinate geoCenter;
          fromENU(origin, center, geoCenter);
          pData->tileCenterPoints.append(QVariant::fromValue(geoCenter));
Valentin Platzgummer's avatar
Valentin Platzgummer committed
276 277
        }
      }
278
      pData->moveToThread(th);
279
#ifdef SNAKE_SHOW_TIME
280 281 282 283 284 285
      qDebug() << "WimaMeasurementArea::doUpdate concurrent update execution "
                  "time: "
               << std::chrono::duration_cast<std::chrono::milliseconds>(
                      std::chrono::high_resolution_clock::now() - start)
                      .count()
               << " ms";
286
#endif
287
      return pData;
288 289 290
    }); // QtConcurrent::run()

    this->_watcher.setFuture(future);
291
  }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
292 293 294 295 296 297 298
#ifdef SNAKE_SHOW_TIME
  qDebug() << "WimaMeasurementArea::doUpdate execution time: "
           << std::chrono::duration_cast<std::chrono::milliseconds>(
                  std::chrono::high_resolution_clock::now() - start)
                  .count()
           << " ms";
#endif
Valentin Platzgummer's avatar
Valentin Platzgummer committed
299 300
}

Valentin Platzgummer's avatar
Valentin Platzgummer committed
301 302 303
void WimaMeasurementArea::deferUpdate() {
  if (this->_timer.isActive()) {
    this->_timer.stop();
304
  }
305 306
  if (this->_tileData.size() > 0) {
    this->_tileData.clear();
307 308
    emit this->tilesChanged();
  }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
309
  this->_timer.start(100);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
310 311
}

312 313 314 315
void WimaMeasurementArea::storeTiles() {
#ifdef SNAKE_SHOW_TIME
  auto start = std::chrono::high_resolution_clock::now();
#endif
316
  this->_tileData = *this->_watcher.result();
317
  this->_calculating = false;
318 319
  // This is expensive. Drawing tiles is expensive too.
  emit this->tilesChanged();
320 321 322 323 324 325 326 327 328
#ifdef SNAKE_SHOW_TIME
  qDebug() << "WimaMeasurementArea::storeTiles() execution time: "
           << std::chrono::duration_cast<std::chrono::milliseconds>(
                  std::chrono::high_resolution_clock::now() - start)
                  .count()
           << " ms";
#endif
}

329 330
void WimaMeasurementArea::init() {
  this->setObjectName(WimaMeasurementAreaName);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
331 332 333 334 335 336 337 338 339 340 341
  connect(&this->_tileHeight, &Fact::rawValueChanged, this,
          &WimaMeasurementArea::deferUpdate);
  connect(&this->_tileWidth, &Fact::rawValueChanged, this,
          &WimaMeasurementArea::deferUpdate);
  connect(&this->_minTileArea, &Fact::rawValueChanged, this,
          &WimaMeasurementArea::deferUpdate);
  connect(this, &WimaArea::pathChanged, this,
          &WimaMeasurementArea::deferUpdate);
  this->_timer.setSingleShot(true);
  connect(&this->_timer, &QTimer::timeout, this,
          &WimaMeasurementArea::doUpdate);
342 343 344
  connect(&this->_watcher,
          &QFutureWatcher<std::unique_ptr<QmlObjectListModel>>::finished, this,
          &WimaMeasurementArea::storeTiles);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
345 346
}

347 348
/*!
 * \class WimaMeasurementArea
349 350
 * \brief Class defining the area inside which the actual drone measurements
 * are performed.
351 352 353
 *
 * \sa WimaArea, WimaController, WimaPlaner
 */