snake.h 6.54 KB
Newer Older
1 2 3
#pragma once

#include <array>
Valentin Platzgummer's avatar
Valentin Platzgummer committed
4
#include <memory>
5 6 7 8 9 10 11 12 13
#include <string>
#include <vector>

#include <boost/geometry.hpp>
#include <boost/units/systems/angle/degrees.hpp>
#include <boost/units/systems/si.hpp>
#include <boost/units/systems/si/io.hpp>
#include <boost/units/systems/si/plane_angle.hpp>
#include <boost/units/systems/si/prefixes.hpp>
14

15 16
namespace bg = boost::geometry;
namespace bu = boost::units;
17

Valentin Platzgummer's avatar
Valentin Platzgummer committed
18 19 20
#include <GeographicLib/Geocentric.hpp>
#include <GeographicLib/LocalCartesian.hpp>

21 22 23
using namespace std;

namespace snake {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
24 25 26
//=========================================================================
// Geometry stuff.
//=========================================================================
27 28 29 30 31 32 33 34 35

typedef bg::model::point<double, 2, bg::cs::cartesian> BoostPoint;
// typedef std::vector<BoostPoint>                        BoostPointList;
typedef bg::model::polygon<BoostPoint> BoostPolygon;
typedef bg::model::linestring<BoostPoint> BoostLineString;
typedef std::vector<std::vector<int64_t>> Int64Matrix;
typedef bg::model::box<BoostPoint> BoostBox;

template <class T> class Matrix {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
36
public:
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
  Matrix() : _elements(0), _m(0), _n(0), _isInitialized(false) {}
  Matrix(size_t m, size_t n) : Matrix(m, n, T{0}) {}
  Matrix(size_t m, size_t n, T value)
      : _elements(m * n), _m(m), _n(n), _isInitialized(true) {
    assert((m > 0) || (n > 0));
    _matrix.resize(_elements, value);
  }

  double get(size_t i, size_t j) const {
    assert(_isInitialized);
    assert(i < _m);
    assert(j < _n);
    return _matrix[i * _m + j];
  }

  size_t getM() const { return _m; }
  size_t getN() const { return _n; }

  void set(size_t i, size_t j, const T &value) {
    assert(_isInitialized);
    assert(i < _m);
    assert(j < _n);
    _matrix[i * _m + j] = value;
  }

  void setDimension(size_t m, size_t n, const T &value) {
    assert((m > 0) || (n > 0));
    assert(!_isInitialized);
    _m = m;
    _n = n;
    _elements = n * m;
    _matrix.resize(_elements, value);
    _isInitialized = true;
  }

  void setDimension(size_t m, size_t n) { setDimension(m, n, T{0}); }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
73 74

private:
75 76 77 78 79
  size_t _elements;
  size_t _m;
  size_t _n;
  bool _isInitialized;
  std::vector<T> _matrix;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
80 81
};

82 83
struct BoundingBox {
  BoundingBox();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
84

85
  void clear();
Valentin Platzgummer's avatar
Valentin Platzgummer committed
86

87 88 89 90
  double width;
  double height;
  double angle;
  BoostPolygon corners;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
91
};
Valentin Platzgummer's avatar
Valentin Platzgummer committed
92

93 94 95 96 97 98 99
template <class GeoPoint>
void toENU(const GeoPoint &origin, const GeoPoint &in, BoostPoint &out) {
  GeographicLib::Geocentric earth(GeographicLib::Constants::WGS84_a(),
                                  GeographicLib::Constants::WGS84_f());
  GeographicLib::LocalCartesian proj(origin.latitude(), origin.longitude(),
                                     origin.altitude(), earth);

Valentin Platzgummer's avatar
Valentin Platzgummer committed
100
  double x = 0, y = 0, z = 0;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
101 102 103
  auto alt = in.altitude();
  alt = std::isnan(alt) ? 0 : alt;
  proj.Forward(in.latitude(), in.longitude(), alt, x, y, z);
104
  out.set<0>(x);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
105
  out.set<1>(y);
106
  (void)z;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
107
}
Valentin Platzgummer's avatar
Valentin Platzgummer committed
108

109 110 111 112 113 114 115
template <class GeoPoint>
void fromENU(const GeoPoint &origin, const BoostPoint &in, GeoPoint &out) {
  GeographicLib::Geocentric earth(GeographicLib::Constants::WGS84_a(),
                                  GeographicLib::Constants::WGS84_f());
  GeographicLib::LocalCartesian proj(origin.latitude(), origin.longitude(),
                                     origin.altitude(), earth);

Valentin Platzgummer's avatar
Valentin Platzgummer committed
116
  double lat = 0, lon = 0, alt = 0;
117 118 119 120
  proj.Reverse(in.get<0>(), in.get<1>(), 0.0, lat, lon, alt);
  out.setLatitude(lat);
  out.setLongitude(lon);
  out.setAltitude(alt);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
121 122
}

123 124 125 126 127 128 129 130 131
template <class GeoPoint, template <class, class...> class Container>
void areaToEnu(const GeoPoint &origin, const Container<GeoPoint> &in,
               BoostPolygon &out) {
  for (auto vertex : in) {
    BoostPoint p;
    toENU(origin, vertex, p);
    out.outer().push_back(p);
  }
  bg::correct(out);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
132 133
}

134 135 136 137 138 139 140 141
template <class GeoPoint, template <class, class...> class Container>
void areaFromEnu(const GeoPoint &origin, BoostPolygon &in,
                 const Container<GeoPoint> &out) {
  for (auto vertex : in.outer()) {
    GeoPoint p;
    fromENU(origin, vertex, p);
    out.push_back(p);
  }
Valentin Platzgummer's avatar
Valentin Platzgummer committed
142
}
143

Valentin Platzgummer's avatar
Valentin Platzgummer committed
144 145 146 147 148 149 150 151 152 153 154 155 156
void polygonCenter(const BoostPolygon &polygon, BoostPoint &center);
bool minimalBoundingBox(const BoostPolygon &polygon, BoundingBox &minBBox);
void offsetPolygon(const BoostPolygon &polygon, BoostPolygon &polygonOffset,
                   double offset);
void graphFromPolygon(const BoostPolygon &polygon,
                      const BoostLineString &vertices, Matrix<double> &graph);
void toDistanceMatrix(Matrix<double> &graph);
bool dijkstraAlgorithm(
    const size_t numElements, size_t startIndex, size_t endIndex,
    std::vector<size_t> &elementPath,
    std::function<double(const size_t, const size_t)> distanceDij);
void shortestPathFromGraph(const Matrix<double> &graph, size_t startIndex,
                           size_t endIndex, std::vector<size_t> &pathIdx);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
157

Valentin Platzgummer's avatar
Valentin Platzgummer committed
158 159 160
typedef bu::quantity<bu::si::length> Length;
typedef bu::quantity<bu::si::area> Area;
typedef bu::quantity<bu::si::plane_angle> Angle;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
161 162
typedef bu::quantity<bu::si::plane_angle> Radian;
typedef bu::quantity<bu::degree::plane_angle> Degree;
Valentin Platzgummer's avatar
Valentin Platzgummer committed
163

Valentin Platzgummer's avatar
Valentin Platzgummer committed
164 165 166 167 168 169 170
bool joinedArea(const std::vector<BoostPolygon *> &areas, BoostPolygon &jArea);
bool joinedArea(const BoostPolygon &mArea, const BoostPolygon &sArea,
                const BoostPolygon &corridor, BoostPolygon &jArea,
                std::string &errorString);
bool tiles(const BoostPolygon &area, Length tileHeight, Length tileWidth,
           Area minTileArea, std::vector<BoostPolygon> &tiles,
           BoundingBox &bbox, std::string &errorString);
171 172 173

using Transects = vector<BoostLineString>;
using Progress = vector<int>;
174
using Route = BoostLineString;
175 176

bool transectsFromScenario(Length distance, Length minLength, Angle angle,
Valentin Platzgummer's avatar
Valentin Platzgummer committed
177 178 179 180
                           const BoostPolygon &mArea,
                           const std::vector<BoostPolygon> &tiles,
                           const Progress &p, Transects &t,
                           string &errorString);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
181 182 183 184 185 186

struct TransectInfo {
  TransectInfo(size_t n, bool r) : index(n), reversed(r) {}
  size_t index;
  bool reversed;
};
187
bool route(const BoostPolygon &area, const Transects &transects,
Valentin Platzgummer's avatar
Valentin Platzgummer committed
188 189
           std::vector<TransectInfo> &transectInfo, Route &route,
           string &errorString);
Valentin Platzgummer's avatar
Valentin Platzgummer committed
190 191

namespace detail {
192
const double offsetConstant =
Valentin Platzgummer's avatar
Valentin Platzgummer committed
193
    0.1; // meter, polygon offset to compenstate for numerical inaccurracies.
Valentin Platzgummer's avatar
Valentin Platzgummer committed
194
} // namespace detail
195 196 197 198 199 200 201 202 203
} // namespace snake

// operator== and operator!= for boost point
namespace boost {
namespace geometry {
namespace model {

bool operator==(snake::BoostPoint p1, snake::BoostPoint p2);
bool operator!=(snake::BoostPoint p1, snake::BoostPoint p2);
204

205 206 207
} // namespace model
} // namespace geometry
} // namespace boost