osrm-backend/src/engine/polyline_compressor.cpp

123 lines
3.3 KiB
C++
Raw Normal View History

2016-01-02 11:13:44 -05:00
#include "engine/polyline_compressor.hpp"
2014-11-28 04:07:06 -05:00
2016-01-28 10:28:44 -05:00
#include <boost/assert.hpp>
#include <cstddef>
#include <cstdlib>
#include <cmath>
2016-01-28 10:28:44 -05:00
#include <algorithm>
2014-11-28 04:07:06 -05:00
2016-01-05 10:51:13 -05:00
namespace osrm
{
namespace engine
{
namespace /*detail*/ // anonymous to keep TU local
{
std::string encode(int number_to_encode)
{
std::string output;
while (number_to_encode >= 0x20)
{
const int next_value = (0x20 | (number_to_encode & 0x1f)) + 63;
output += static_cast<char>(next_value);
number_to_encode >>= 5;
}
number_to_encode += 63;
output += static_cast<char>(number_to_encode);
return output;
}
2016-01-05 10:51:13 -05:00
std::string encode(std::vector<int> &numbers)
2014-11-28 04:07:06 -05:00
{
std::string output;
for (auto &number : numbers)
2014-11-28 04:07:06 -05:00
{
bool isNegative = number < 0;
if (isNegative)
{
const unsigned binary = std::llabs(number);
const unsigned twos = (~binary) + 1u;
number = twos;
}
number <<= 1u;
if (isNegative)
2014-11-28 04:07:06 -05:00
{
number = ~number;
2014-11-28 04:07:06 -05:00
}
}
for (const int number : numbers)
{
output += encode(number);
2014-11-28 04:07:06 -05:00
}
return output;
}
} // anonymous ns
2014-11-28 04:07:06 -05:00
2016-01-28 10:28:44 -05:00
std::string encodePolyline(CoordVectorForwardIter begin, CoordVectorForwardIter end)
2014-11-28 04:07:06 -05:00
{
2016-01-28 10:28:44 -05:00
auto size = std::distance(begin, end);
if (size == 0)
2014-11-28 04:07:06 -05:00
{
return {};
}
std::vector<int> delta_numbers;
2016-01-28 10:28:44 -05:00
BOOST_ASSERT(size > 0);
delta_numbers.reserve((size - 1) * 2);
util::Coordinate previous_coordinate{util::FixedLongitude(0), util::FixedLatitude(0)};
std::for_each(begin, end, [&delta_numbers, &previous_coordinate](const util::Coordinate loc)
{
const int lat_diff = static_cast<int>(loc.lat - previous_coordinate.lat) *
detail::COORDINATE_TO_POLYLINE;
const int lon_diff = static_cast<int>(loc.lon - previous_coordinate.lon) *
detail::COORDINATE_TO_POLYLINE;
delta_numbers.emplace_back(lat_diff);
delta_numbers.emplace_back(lon_diff);
previous_coordinate = loc;
});
return encode(delta_numbers);
2014-11-28 04:07:06 -05:00
}
std::vector<util::Coordinate> decodePolyline(const std::string &geometry_string)
2015-05-31 07:57:27 -04:00
{
std::vector<util::Coordinate> new_coordinates;
2015-05-31 07:57:27 -04:00
int index = 0, len = geometry_string.size();
int lat = 0, lng = 0;
2016-01-05 06:04:04 -05:00
while (index < len)
2015-05-31 07:57:27 -04:00
{
int b, shift = 0, result = 0;
2016-01-05 06:04:04 -05:00
do
2015-05-31 07:57:27 -04:00
{
b = geometry_string.at(index++) - 63;
result |= (b & 0x1f) << shift;
shift += 5;
} while (b >= 0x20);
int dlat = ((result & 1) != 0 ? ~(result >> 1) : (result >> 1));
lat += dlat;
shift = 0;
result = 0;
2016-01-05 06:04:04 -05:00
do
2015-05-31 07:57:27 -04:00
{
b = geometry_string.at(index++) - 63;
result |= (b & 0x1f) << shift;
shift += 5;
} while (b >= 0x20);
int dlng = ((result & 1) != 0 ? ~(result >> 1) : (result >> 1));
lng += dlng;
util::Coordinate p;
p.lat = util::FixedLatitude(lat * detail::POLYLINE_TO_COORDINATE);
p.lon = util::FixedLongitude(lng * detail::POLYLINE_TO_COORDINATE);
2015-05-31 07:57:27 -04:00
new_coordinates.push_back(p);
}
return new_coordinates;
}
2016-01-05 10:51:13 -05:00
}
}