Implementation of border conditions cases like zero on one route point support.

This commit is contained in:
Vladimir Byko-Ianko 2016-09-08 15:07:46 +03:00
parent f29e63de51
commit 14be10af19
2 changed files with 38 additions and 20 deletions

View file

@ -57,9 +57,9 @@ bool NormalizeChartData(vector<double> const & distanceDataM,
{
double constexpr kEpsilon = 1e-6;
if (distanceDataM.empty() || resultPointCount == 0 || resultPointCount == 1 ||
distanceDataM.size() != altitudeDataM.size())
if (distanceDataM.size() != altitudeDataM.size())
{
LOG(LERROR, ("Altitude and distance data have different size."));
return false;
}
@ -69,8 +69,11 @@ bool NormalizeChartData(vector<double> const & distanceDataM,
return false;
}
double const routeLenM = distanceDataM.back();
double const stepLenM = routeLenM / static_cast<double>(resultPointCount - 1);
if (distanceDataM.empty() || resultPointCount == 0)
{
uniformAltitudeDataM.clear();
return true;
}
auto const calculateAltitude = [&](double distFormStartM) {
if (distFormStartM <= distanceDataM.front())
@ -97,19 +100,27 @@ bool NormalizeChartData(vector<double> const & distanceDataM,
k * (distFormStartM - distanceDataM[prevPointIdx]);
};
double const routeLenM = distanceDataM.back();
uniformAltitudeDataM.resize(resultPointCount);
if (resultPointCount == 1)
{
uniformAltitudeDataM[0] = calculateAltitude(routeLenM / 2.0);
return true;
}
double const stepLenM = routeLenM / static_cast<double>(resultPointCount - 1);
for (size_t i = 0; i < resultPointCount; ++i)
uniformAltitudeDataM[i] = calculateAltitude(static_cast<double>(i) * stepLenM);
return true;
}
void GenerateYAxisChartData(uint32_t height, double minMetersPerPxl,
bool GenerateYAxisChartData(uint32_t height, double minMetersPerPxl,
vector<double> const & altitudeDataM, vector<double> & yAxisDataPxl)
{
yAxisDataPxl.clear();
if (altitudeDataM.empty())
return;
return true;
uint32_t constexpr kHeightIndentPxl = 2;
uint32_t heightIndent = kHeightIndentPxl;
@ -128,20 +139,22 @@ void GenerateYAxisChartData(uint32_t height, double minMetersPerPxl,
if (meterPerPxl == 0.0)
{
LOG(LERROR, ("meterPerPxl == 0.0"));
return;
return false;
}
size_t const altitudeDataSz = altitudeDataM.size();
yAxisDataPxl.resize(altitudeDataSz);
for (size_t i = 0; i < altitudeDataSz; ++i)
yAxisDataPxl[i] = height - heightIndent - (altitudeDataM[i] - minAltM) / meterPerPxl;
return true;
}
void GenerateChartByPoints(uint32_t width, uint32_t height, vector<m2::PointD> const & geometry,
bool lightTheme, vector<uint8_t> & frameBuffer)
{
frameBuffer.clear();
if (width == 0 || height == 0 || geometry.empty())
if (width == 0 || height == 0)
return;
agg::rgba8 const kBackgroundColor = agg::rgba8(255, 255, 255, 0);
@ -166,8 +179,10 @@ void GenerateChartByPoints(uint32_t width, uint32_t height, vector<m2::PointD> c
frameBuffer.assign(width * kBPP * height, 0);
renderBuffer.attach(&frameBuffer[0], static_cast<unsigned>(width),
static_cast<unsigned>(height), static_cast<int>(width * kBPP));
// Background.
baseRenderer.reset_clipping(true);
unsigned op = pixelFormat.comp_op();
unsigned const op = pixelFormat.comp_op();
pixelFormat.comp_op(agg::comp_op_src);
baseRenderer.clear(kBackgroundColor);
pixelFormat.comp_op(op);
@ -175,6 +190,9 @@ void GenerateChartByPoints(uint32_t width, uint32_t height, vector<m2::PointD> c
agg::rasterizer_scanline_aa<> rasterizer;
rasterizer.clip_box(0, 0, width, height);
if (geometry.empty())
return; /* No chart line to draw. */
// Polygon under chart line.
agg::path_storage underChartGeometryPath;
underChartGeometryPath.move_to(geometry.front().x, static_cast<double>(height));
@ -210,24 +228,24 @@ bool GenerateChart(uint32_t width, uint32_t height, vector<double> const & dista
return false;
}
if (altitudeDataM.empty())
return false;
vector<double> uniformAltitudeDataM;
if (!NormalizeChartData(distanceDataM, altitudeDataM, width, uniformAltitudeDataM))
return false;
vector<double> yAxisDataPxl;
GenerateYAxisChartData(height, 1.0 /* minMetersPerPxl */, uniformAltitudeDataM, yAxisDataPxl);
size_t const uniformAltitudeDataSize = yAxisDataPxl.size();
if (uniformAltitudeDataSize <= 1)
if (!GenerateYAxisChartData(height, 1.0 /* minMetersPerPxl */, uniformAltitudeDataM, yAxisDataPxl))
return false;
double const oneSegLenPix = static_cast<double>(width) / (uniformAltitudeDataSize - 1);
size_t const uniformAltitudeDataSize = yAxisDataPxl.size();
vector<m2::PointD> geometry(uniformAltitudeDataSize);
for (size_t i = 0; i < uniformAltitudeDataSize; ++i)
geometry[i] = m2::PointD(i * oneSegLenPix, yAxisDataPxl[i]);
if (uniformAltitudeDataSize != 0)
{
double const oneSegLenPix =
static_cast<double>(width) / (uniformAltitudeDataSize == 1 ? 1 : (uniformAltitudeDataSize - 1));
for (size_t i = 0; i < uniformAltitudeDataSize; ++i)
geometry[i] = m2::PointD(i * oneSegLenPix, yAxisDataPxl[i]);
}
frameBuffer.clear();
GenerateChartByPoints(width, height, geometry, lightTheme, frameBuffer);

View file

@ -22,7 +22,7 @@ bool NormalizeChartData(vector<double> const & distanceDataM,
/// \param minMetersInPixel minimum meter number per height pixel.
/// \param altitudeDataM altitude data vector in meters.
/// \param yAxisDataPxl Y-axis data of altitude chart in pixels.
void GenerateYAxisChartData(uint32_t height, double minMetersPerPxl,
bool GenerateYAxisChartData(uint32_t height, double minMetersPerPxl,
vector<double> const & altitudeDataM, vector<double> & yAxisDataPxl);
/// \brief generates chart image on a canvas with size |width|, |height| with |geometry|.