532 lines
18 KiB
C++
532 lines
18 KiB
C++
// Copyright 2018 Slightech Co., Ltd. All rights reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
#include "gpl.h"
|
|
|
|
#include <set>
|
|
#ifdef _WIN32
|
|
#include <winsock.h>
|
|
#define M_PI (3.14159265358979323846)
|
|
#else
|
|
#include <time.h>
|
|
#endif
|
|
|
|
// source:
|
|
// https://stackoverflow.com/questions/5167269/clock-gettime-alternative-in-mac-os-x
|
|
#ifdef __APPLE__
|
|
#include <mach/mach_time.h>
|
|
#define ORWL_NANO (+1.0E-9)
|
|
#define ORWL_GIGA UINT64_C(1000000000)
|
|
|
|
static double orwl_timebase = 0.0;
|
|
static uint64_t orwl_timestart = 0;
|
|
|
|
struct timespec orwl_gettime(void) {
|
|
// be more careful in a multithreaded environement
|
|
if (!orwl_timestart) {
|
|
mach_timebase_info_data_t tb = {0};
|
|
mach_timebase_info(&tb);
|
|
orwl_timebase = tb.numer;
|
|
orwl_timebase /= tb.denom;
|
|
orwl_timestart = mach_absolute_time();
|
|
}
|
|
struct timespec t;
|
|
double diff = (mach_absolute_time() - orwl_timestart) * orwl_timebase;
|
|
t.tv_sec = diff * ORWL_NANO;
|
|
t.tv_nsec = diff - (t.tv_sec * ORWL_GIGA);
|
|
return t;
|
|
}
|
|
#endif // __APPLE__
|
|
|
|
const double WGS84_A = 6378137.0;
|
|
const double WGS84_ECCSQ = 0.00669437999013;
|
|
|
|
// Windows lacks fminf
|
|
#ifndef fminf
|
|
#define fminf(x, y) (((x) < (y)) ? (x) : (y))
|
|
#endif
|
|
|
|
MYNTEYE_BEGIN_NAMESPACE
|
|
|
|
namespace models {
|
|
|
|
double hypot3(double x, double y, double z) {
|
|
return sqrt(square(x) + square(y) + square(z));
|
|
}
|
|
|
|
float hypot3f(float x, float y, float z) {
|
|
return sqrtf(square(x) + square(y) + square(z));
|
|
}
|
|
|
|
double d2r(double deg) {
|
|
return deg / 180.0 * M_PI;
|
|
}
|
|
|
|
float d2r(float deg) {
|
|
return deg / 180.0f * M_PI;
|
|
}
|
|
|
|
double r2d(double rad) {
|
|
return rad / M_PI * 180.0;
|
|
}
|
|
|
|
float r2d(float rad) {
|
|
return rad / M_PI * 180.0f;
|
|
}
|
|
|
|
double sinc(double theta) {
|
|
return sin(theta) / theta;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
#include <sys/timeb.h>
|
|
#include <sys/types.h>
|
|
#include <winsock.h>
|
|
LARGE_INTEGER
|
|
getFILETIMEoffset() {
|
|
SYSTEMTIME s;
|
|
FILETIME f;
|
|
LARGE_INTEGER t;
|
|
|
|
s.wYear = 1970;
|
|
s.wMonth = 1;
|
|
s.wDay = 1;
|
|
s.wHour = 0;
|
|
s.wMinute = 0;
|
|
s.wSecond = 0;
|
|
s.wMilliseconds = 0;
|
|
SystemTimeToFileTime(&s, &f);
|
|
t.QuadPart = f.dwHighDateTime;
|
|
t.QuadPart <<= 32;
|
|
t.QuadPart |= f.dwLowDateTime;
|
|
return (t);
|
|
}
|
|
|
|
#endif
|
|
|
|
float colormapAutumn[128][3] = {
|
|
{1.0f, 0.f, 0.f}, {1.0f, 0.007874f, 0.f}, {1.0f, 0.015748f, 0.f},
|
|
{1.0f, 0.023622f, 0.f}, {1.0f, 0.031496f, 0.f}, {1.0f, 0.03937f, 0.f},
|
|
{1.0f, 0.047244f, 0.f}, {1.0f, 0.055118f, 0.f}, {1.0f, 0.062992f, 0.f},
|
|
{1.0f, 0.070866f, 0.f}, {1.0f, 0.07874f, 0.f}, {1.0f, 0.086614f, 0.f},
|
|
{1.0f, 0.094488f, 0.f}, {1.0f, 0.10236f, 0.f}, {1.0f, 0.11024f, 0.f},
|
|
{1.0f, 0.11811f, 0.f}, {1.0f, 0.12598f, 0.f}, {1.0f, 0.13386f, 0.f},
|
|
{1.0f, 0.14173f, 0.f}, {1.0f, 0.14961f, 0.f}, {1.0f, 0.15748f, 0.f},
|
|
{1.0f, 0.16535f, 0.f}, {1.0f, 0.17323f, 0.f}, {1.0f, 0.1811f, 0.f},
|
|
{1.0f, 0.18898f, 0.f}, {1.0f, 0.19685f, 0.f}, {1.0f, 0.20472f, 0.f},
|
|
{1.0f, 0.2126f, 0.f}, {1.0f, 0.22047f, 0.f}, {1.0f, 0.22835f, 0.f},
|
|
{1.0f, 0.23622f, 0.f}, {1.0f, 0.24409f, 0.f}, {1.0f, 0.25197f, 0.f},
|
|
{1.0f, 0.25984f, 0.f}, {1.0f, 0.26772f, 0.f}, {1.0f, 0.27559f, 0.f},
|
|
{1.0f, 0.28346f, 0.f}, {1.0f, 0.29134f, 0.f}, {1.0f, 0.29921f, 0.f},
|
|
{1.0f, 0.30709f, 0.f}, {1.0f, 0.31496f, 0.f}, {1.0f, 0.32283f, 0.f},
|
|
{1.0f, 0.33071f, 0.f}, {1.0f, 0.33858f, 0.f}, {1.0f, 0.34646f, 0.f},
|
|
{1.0f, 0.35433f, 0.f}, {1.0f, 0.3622f, 0.f}, {1.0f, 0.37008f, 0.f},
|
|
{1.0f, 0.37795f, 0.f}, {1.0f, 0.38583f, 0.f}, {1.0f, 0.3937f, 0.f},
|
|
{1.0f, 0.40157f, 0.f}, {1.0f, 0.40945f, 0.f}, {1.0f, 0.41732f, 0.f},
|
|
{1.0f, 0.4252f, 0.f}, {1.0f, 0.43307f, 0.f}, {1.0f, 0.44094f, 0.f},
|
|
{1.0f, 0.44882f, 0.f}, {1.0f, 0.45669f, 0.f}, {1.0f, 0.46457f, 0.f},
|
|
{1.0f, 0.47244f, 0.f}, {1.0f, 0.48031f, 0.f}, {1.0f, 0.48819f, 0.f},
|
|
{1.0f, 0.49606f, 0.f}, {1.0f, 0.50394f, 0.f}, {1.0f, 0.51181f, 0.f},
|
|
{1.0f, 0.51969f, 0.f}, {1.0f, 0.52756f, 0.f}, {1.0f, 0.53543f, 0.f},
|
|
{1.0f, 0.54331f, 0.f}, {1.0f, 0.55118f, 0.f}, {1.0f, 0.55906f, 0.f},
|
|
{1.0f, 0.56693f, 0.f}, {1.0f, 0.5748f, 0.f}, {1.0f, 0.58268f, 0.f},
|
|
{1.0f, 0.59055f, 0.f}, {1.0f, 0.59843f, 0.f}, {1.0f, 0.6063f, 0.f},
|
|
{1.0f, 0.61417f, 0.f}, {1.0f, 0.62205f, 0.f}, {1.0f, 0.62992f, 0.f},
|
|
{1.0f, 0.6378f, 0.f}, {1.0f, 0.64567f, 0.f}, {1.0f, 0.65354f, 0.f},
|
|
{1.0f, 0.66142f, 0.f}, {1.0f, 0.66929f, 0.f}, {1.0f, 0.67717f, 0.f},
|
|
{1.0f, 0.68504f, 0.f}, {1.0f, 0.69291f, 0.f}, {1.0f, 0.70079f, 0.f},
|
|
{1.0f, 0.70866f, 0.f}, {1.0f, 0.71654f, 0.f}, {1.0f, 0.72441f, 0.f},
|
|
{1.0f, 0.73228f, 0.f}, {1.0f, 0.74016f, 0.f}, {1.0f, 0.74803f, 0.f},
|
|
{1.0f, 0.75591f, 0.f}, {1.0f, 0.76378f, 0.f}, {1.0f, 0.77165f, 0.f},
|
|
{1.0f, 0.77953f, 0.f}, {1.0f, 0.7874f, 0.f}, {1.0f, 0.79528f, 0.f},
|
|
{1.0f, 0.80315f, 0.f}, {1.0f, 0.81102f, 0.f}, {1.0f, 0.8189f, 0.f},
|
|
{1.0f, 0.82677f, 0.f}, {1.0f, 0.83465f, 0.f}, {1.0f, 0.84252f, 0.f},
|
|
{1.0f, 0.85039f, 0.f}, {1.0f, 0.85827f, 0.f}, {1.0f, 0.86614f, 0.f},
|
|
{1.0f, 0.87402f, 0.f}, {1.0f, 0.88189f, 0.f}, {1.0f, 0.88976f, 0.f},
|
|
{1.0f, 0.89764f, 0.f}, {1.0f, 0.90551f, 0.f}, {1.0f, 0.91339f, 0.f},
|
|
{1.0f, 0.92126f, 0.f}, {1.0f, 0.92913f, 0.f}, {1.0f, 0.93701f, 0.f},
|
|
{1.0f, 0.94488f, 0.f}, {1.0f, 0.95276f, 0.f}, {1.0f, 0.96063f, 0.f},
|
|
{1.0f, 0.9685f, 0.f}, {1.0f, 0.97638f, 0.f}, {1.0f, 0.98425f, 0.f},
|
|
{1.0f, 0.99213f, 0.f}, {1.0f, 1.0f, 0.0f}};
|
|
|
|
float colormapJet[128][3] = {
|
|
{0.0f, 0.0f, 0.53125f}, {0.0f, 0.0f, 0.5625f},
|
|
{0.0f, 0.0f, 0.59375f}, {0.0f, 0.0f, 0.625f},
|
|
{0.0f, 0.0f, 0.65625f}, {0.0f, 0.0f, 0.6875f},
|
|
{0.0f, 0.0f, 0.71875f}, {0.0f, 0.0f, 0.75f},
|
|
{0.0f, 0.0f, 0.78125f}, {0.0f, 0.0f, 0.8125f},
|
|
{0.0f, 0.0f, 0.84375f}, {0.0f, 0.0f, 0.875f},
|
|
{0.0f, 0.0f, 0.90625f}, {0.0f, 0.0f, 0.9375f},
|
|
{0.0f, 0.0f, 0.96875f}, {0.0f, 0.0f, 1.0f},
|
|
{0.0f, 0.03125f, 1.0f}, {0.0f, 0.0625f, 1.0f},
|
|
{0.0f, 0.09375f, 1.0f}, {0.0f, 0.125f, 1.0f},
|
|
{0.0f, 0.15625f, 1.0f}, {0.0f, 0.1875f, 1.0f},
|
|
{0.0f, 0.21875f, 1.0f}, {0.0f, 0.25f, 1.0f},
|
|
{0.0f, 0.28125f, 1.0f}, {0.0f, 0.3125f, 1.0f},
|
|
{0.0f, 0.34375f, 1.0f}, {0.0f, 0.375f, 1.0f},
|
|
{0.0f, 0.40625f, 1.0f}, {0.0f, 0.4375f, 1.0f},
|
|
{0.0f, 0.46875f, 1.0f}, {0.0f, 0.5f, 1.0f},
|
|
{0.0f, 0.53125f, 1.0f}, {0.0f, 0.5625f, 1.0f},
|
|
{0.0f, 0.59375f, 1.0f}, {0.0f, 0.625f, 1.0f},
|
|
{0.0f, 0.65625f, 1.0f}, {0.0f, 0.6875f, 1.0f},
|
|
{0.0f, 0.71875f, 1.0f}, {0.0f, 0.75f, 1.0f},
|
|
{0.0f, 0.78125f, 1.0f}, {0.0f, 0.8125f, 1.0f},
|
|
{0.0f, 0.84375f, 1.0f}, {0.0f, 0.875f, 1.0f},
|
|
{0.0f, 0.90625f, 1.0f}, {0.0f, 0.9375f, 1.0f},
|
|
{0.0f, 0.96875f, 1.0f}, {0.0f, 1.0f, 1.0f},
|
|
{0.03125f, 1.0f, 0.96875f}, {0.0625f, 1.0f, 0.9375f},
|
|
{0.09375f, 1.0f, 0.90625f}, {0.125f, 1.0f, 0.875f},
|
|
{0.15625f, 1.0f, 0.84375f}, {0.1875f, 1.0f, 0.8125f},
|
|
{0.21875f, 1.0f, 0.78125f}, {0.25f, 1.0f, 0.75f},
|
|
{0.28125f, 1.0f, 0.71875f}, {0.3125f, 1.0f, 0.6875f},
|
|
{0.34375f, 1.0f, 0.65625f}, {0.375f, 1.0f, 0.625f},
|
|
{0.40625f, 1.0f, 0.59375f}, {0.4375f, 1.0f, 0.5625f},
|
|
{0.46875f, 1.0f, 0.53125f}, {0.5f, 1.0f, 0.5f},
|
|
{0.53125f, 1.0f, 0.46875f}, {0.5625f, 1.0f, 0.4375f},
|
|
{0.59375f, 1.0f, 0.40625f}, {0.625f, 1.0f, 0.375f},
|
|
{0.65625f, 1.0f, 0.34375f}, {0.6875f, 1.0f, 0.3125f},
|
|
{0.71875f, 1.0f, 0.28125f}, {0.75f, 1.0f, 0.25f},
|
|
{0.78125f, 1.0f, 0.21875f}, {0.8125f, 1.0f, 0.1875f},
|
|
{0.84375f, 1.0f, 0.15625f}, {0.875f, 1.0f, 0.125f},
|
|
{0.90625f, 1.0f, 0.09375f}, {0.9375f, 1.0f, 0.0625f},
|
|
{0.96875f, 1.0f, 0.03125f}, {1.0f, 1.0f, 0.0f},
|
|
{1.0f, 0.96875f, 0.0f}, {1.0f, 0.9375f, 0.0f},
|
|
{1.0f, 0.90625f, 0.0f}, {1.0f, 0.875f, 0.0f},
|
|
{1.0f, 0.84375f, 0.0f}, {1.0f, 0.8125f, 0.0f},
|
|
{1.0f, 0.78125f, 0.0f}, {1.0f, 0.75f, 0.0f},
|
|
{1.0f, 0.71875f, 0.0f}, {1.0f, 0.6875f, 0.0f},
|
|
{1.0f, 0.65625f, 0.0f}, {1.0f, 0.625f, 0.0f},
|
|
{1.0f, 0.59375f, 0.0f}, {1.0f, 0.5625f, 0.0f},
|
|
{1.0f, 0.53125f, 0.0f}, {1.0f, 0.5f, 0.0f},
|
|
{1.0f, 0.46875f, 0.0f}, {1.0f, 0.4375f, 0.0f},
|
|
{1.0f, 0.40625f, 0.0f}, {1.0f, 0.375f, 0.0f},
|
|
{1.0f, 0.34375f, 0.0f}, {1.0f, 0.3125f, 0.0f},
|
|
{1.0f, 0.28125f, 0.0f}, {1.0f, 0.25f, 0.0f},
|
|
{1.0f, 0.21875f, 0.0f}, {1.0f, 0.1875f, 0.0f},
|
|
{1.0f, 0.15625f, 0.0f}, {1.0f, 0.125f, 0.0f},
|
|
{1.0f, 0.09375f, 0.0f}, {1.0f, 0.0625f, 0.0f},
|
|
{1.0f, 0.03125f, 0.0f}, {1.0f, 0.0f, 0.0f},
|
|
{0.96875f, 0.0f, 0.0f}, {0.9375f, 0.0f, 0.0f},
|
|
{0.90625f, 0.0f, 0.0f}, {0.875f, 0.0f, 0.0f},
|
|
{0.84375f, 0.0f, 0.0f}, {0.8125f, 0.0f, 0.0f},
|
|
{0.78125f, 0.0f, 0.0f}, {0.75f, 0.0f, 0.0f},
|
|
{0.71875f, 0.0f, 0.0f}, {0.6875f, 0.0f, 0.0f},
|
|
{0.65625f, 0.0f, 0.0f}, {0.625f, 0.0f, 0.0f},
|
|
{0.59375f, 0.0f, 0.0f}, {0.5625f, 0.0f, 0.0f},
|
|
{0.53125f, 0.0f, 0.0f}, {0.5f, 0.0f, 0.0f}};
|
|
|
|
void colorDepthImage(
|
|
cv::Mat &imgDepth, cv::Mat &imgColoredDepth, float minRange, // NOLINT
|
|
float maxRange) {
|
|
imgColoredDepth = cv::Mat::zeros(imgDepth.size(), CV_8UC3);
|
|
|
|
for (int i = 0; i < imgColoredDepth.rows; ++i) {
|
|
const float *depth = imgDepth.ptr<float>(i);
|
|
unsigned char *pixel = imgColoredDepth.ptr<unsigned char>(i);
|
|
for (int j = 0; j < imgColoredDepth.cols; ++j) {
|
|
if (depth[j] != 0) {
|
|
int idx = fminf(depth[j] - minRange, maxRange - minRange) /
|
|
(maxRange - minRange) * 127.0f;
|
|
idx = 127 - idx;
|
|
|
|
pixel[0] = colormapJet[idx][2] * 255.0f;
|
|
pixel[1] = colormapJet[idx][1] * 255.0f;
|
|
pixel[2] = colormapJet[idx][0] * 255.0f;
|
|
}
|
|
|
|
pixel += 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool colormap(
|
|
const std::string &name, unsigned char idx, float &r, float &g, float &b) { // NOLINT
|
|
if (name.compare("jet") == 0) {
|
|
float *color = colormapJet[idx];
|
|
|
|
r = color[0];
|
|
g = color[1];
|
|
b = color[2];
|
|
|
|
return true;
|
|
} else if (name.compare("autumn") == 0) {
|
|
float *color = colormapAutumn[idx];
|
|
|
|
r = color[0];
|
|
g = color[1];
|
|
b = color[2];
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
std::vector<cv::Point2i> bresLine(int x0, int y0, int x1, int y1) {
|
|
// Bresenham's line algorithm
|
|
// Find cells intersected by line between (x0,y0) and (x1,y1)
|
|
|
|
std::vector<cv::Point2i> cells;
|
|
|
|
int dx = std::abs(x1 - x0);
|
|
int dy = std::abs(y1 - y0);
|
|
|
|
int sx = (x0 < x1) ? 1 : -1;
|
|
int sy = (y0 < y1) ? 1 : -1;
|
|
|
|
int err = dx - dy;
|
|
|
|
while (1) {
|
|
cells.push_back(cv::Point2i(x0, y0));
|
|
|
|
if (x0 == x1 && y0 == y1) {
|
|
break;
|
|
}
|
|
|
|
int e2 = 2 * err;
|
|
if (e2 > -dy) {
|
|
err -= dy;
|
|
x0 += sx;
|
|
}
|
|
if (e2 < dx) {
|
|
err += dx;
|
|
y0 += sy;
|
|
}
|
|
}
|
|
|
|
return cells;
|
|
}
|
|
|
|
std::vector<cv::Point2i> bresCircle(int x0, int y0, int r) {
|
|
// Bresenham's circle algorithm
|
|
// Find cells intersected by circle with center (x0,y0) and radius r
|
|
|
|
std::vector<std::vector<bool> > mask(2 * r + 1);
|
|
|
|
for (int i = 0; i < 2 * r + 1; ++i) {
|
|
mask[i].resize(2 * r + 1);
|
|
for (int j = 0; j < 2 * r + 1; ++j) {
|
|
mask[i][j] = false;
|
|
}
|
|
}
|
|
|
|
int f = 1 - r;
|
|
int ddF_x = 1;
|
|
int ddF_y = -2 * r;
|
|
int x = 0;
|
|
int y = r;
|
|
|
|
std::vector<cv::Point2i> line;
|
|
|
|
line = bresLine(x0, y0 - r, x0, y0 + r);
|
|
for (std::vector<cv::Point2i>::iterator it = line.begin(); it != line.end();
|
|
++it) {
|
|
mask[it->x - x0 + r][it->y - y0 + r] = true;
|
|
}
|
|
|
|
line = bresLine(x0 - r, y0, x0 + r, y0);
|
|
for (std::vector<cv::Point2i>::iterator it = line.begin(); it != line.end();
|
|
++it) {
|
|
mask[it->x - x0 + r][it->y - y0 + r] = true;
|
|
}
|
|
|
|
while (x < y) {
|
|
if (f >= 0) {
|
|
y--;
|
|
ddF_y += 2;
|
|
f += ddF_y;
|
|
}
|
|
|
|
x++;
|
|
ddF_x += 2;
|
|
f += ddF_x;
|
|
|
|
line = bresLine(x0 - x, y0 + y, x0 + x, y0 + y);
|
|
for (std::vector<cv::Point2i>::iterator it = line.begin(); it != line.end();
|
|
++it) {
|
|
mask[it->x - x0 + r][it->y - y0 + r] = true;
|
|
}
|
|
|
|
line = bresLine(x0 - x, y0 - y, x0 + x, y0 - y);
|
|
for (std::vector<cv::Point2i>::iterator it = line.begin(); it != line.end();
|
|
++it) {
|
|
mask[it->x - x0 + r][it->y - y0 + r] = true;
|
|
}
|
|
|
|
line = bresLine(x0 - y, y0 + x, x0 + y, y0 + x);
|
|
for (std::vector<cv::Point2i>::iterator it = line.begin(); it != line.end();
|
|
++it) {
|
|
mask[it->x - x0 + r][it->y - y0 + r] = true;
|
|
}
|
|
|
|
line = bresLine(x0 - y, y0 - x, x0 + y, y0 - x);
|
|
for (std::vector<cv::Point2i>::iterator it = line.begin(); it != line.end();
|
|
++it) {
|
|
mask[it->x - x0 + r][it->y - y0 + r] = true;
|
|
}
|
|
}
|
|
|
|
std::vector<cv::Point2i> cells;
|
|
for (int i = 0; i < 2 * r + 1; ++i) {
|
|
for (int j = 0; j < 2 * r + 1; ++j) {
|
|
if (mask[i][j]) {
|
|
cells.push_back(cv::Point2i(i - r + x0, j - r + y0));
|
|
}
|
|
}
|
|
}
|
|
|
|
return cells;
|
|
}
|
|
|
|
void fitCircle(
|
|
const std::vector<cv::Point2d> &points, double ¢erX, double ¢erY, // NOLINT
|
|
double &radius) { // NOLINT
|
|
// D. Umbach, and K. Jones, A Few Methods for Fitting Circles to Data,
|
|
// IEEE Transactions on Instrumentation and Measurement, 2000
|
|
// We use the modified least squares method.
|
|
double sum_x = 0.0;
|
|
double sum_y = 0.0;
|
|
double sum_xx = 0.0;
|
|
double sum_xy = 0.0;
|
|
double sum_yy = 0.0;
|
|
double sum_xxx = 0.0;
|
|
double sum_xxy = 0.0;
|
|
double sum_xyy = 0.0;
|
|
double sum_yyy = 0.0;
|
|
|
|
int n = points.size();
|
|
for (int i = 0; i < n; ++i) {
|
|
double x = points.at(i).x;
|
|
double y = points.at(i).y;
|
|
|
|
sum_x += x;
|
|
sum_y += y;
|
|
sum_xx += x * x;
|
|
sum_xy += x * y;
|
|
sum_yy += y * y;
|
|
sum_xxx += x * x * x;
|
|
sum_xxy += x * x * y;
|
|
sum_xyy += x * y * y;
|
|
sum_yyy += y * y * y;
|
|
}
|
|
|
|
double A = n * sum_xx - square(sum_x);
|
|
double B = n * sum_xy - sum_x * sum_y;
|
|
double C = n * sum_yy - square(sum_y);
|
|
double D =
|
|
0.5 * (n * sum_xyy - sum_x * sum_yy + n * sum_xxx - sum_x * sum_xx);
|
|
double E =
|
|
0.5 * (n * sum_xxy - sum_y * sum_xx + n * sum_yyy - sum_y * sum_yy);
|
|
|
|
centerX = (D * C - B * E) / (A * C - square(B));
|
|
centerY = (A * E - B * D) / (A * C - square(B));
|
|
|
|
double sum_r = 0.0;
|
|
for (int i = 0; i < n; ++i) {
|
|
double x = points.at(i).x;
|
|
double y = points.at(i).y;
|
|
|
|
sum_r += hypot(x - centerX, y - centerY);
|
|
}
|
|
|
|
radius = sum_r / n;
|
|
}
|
|
|
|
std::vector<cv::Point2d> intersectCircles(
|
|
double x1, double y1, double r1, double x2, double y2, double r2) {
|
|
std::vector<cv::Point2d> ipts;
|
|
|
|
double d = hypot(x1 - x2, y1 - y2);
|
|
if (d > r1 + r2) {
|
|
// circles are separate
|
|
return ipts;
|
|
}
|
|
if (d < fabs(r1 - r2)) {
|
|
// one circle is contained within the other
|
|
return ipts;
|
|
}
|
|
|
|
double a = (square(r1) - square(r2) + square(d)) / (2.0 * d);
|
|
double h = sqrt(square(r1) - square(a));
|
|
|
|
double x3 = x1 + a * (x2 - x1) / d;
|
|
double y3 = y1 + a * (y2 - y1) / d;
|
|
|
|
if (h < 1e-10) {
|
|
// two circles touch at one point
|
|
ipts.push_back(cv::Point2d(x3, y3));
|
|
return ipts;
|
|
}
|
|
|
|
ipts.push_back(cv::Point2d(x3 + h * (y2 - y1) / d, y3 - h * (x2 - x1) / d));
|
|
ipts.push_back(cv::Point2d(x3 - h * (y2 - y1) / d, y3 + h * (x2 - x1) / d));
|
|
return ipts;
|
|
}
|
|
|
|
char UTMLetterDesignator(double latitude) {
|
|
// This routine determines the correct UTM letter designator for the given
|
|
// latitude
|
|
// returns 'Z' if latitude is outside the UTM limits of 84N to 80S
|
|
// Written by Chuck Gantz- chuck.gantz@globalstar.com
|
|
char letterDesignator;
|
|
|
|
if ((84.0 >= latitude) && (latitude >= 72.0))
|
|
letterDesignator = 'X';
|
|
else if ((72.0 > latitude) && (latitude >= 64.0))
|
|
letterDesignator = 'W';
|
|
else if ((64.0 > latitude) && (latitude >= 56.0))
|
|
letterDesignator = 'V';
|
|
else if ((56.0 > latitude) && (latitude >= 48.0))
|
|
letterDesignator = 'U';
|
|
else if ((48.0 > latitude) && (latitude >= 40.0))
|
|
letterDesignator = 'T';
|
|
else if ((40.0 > latitude) && (latitude >= 32.0))
|
|
letterDesignator = 'S';
|
|
else if ((32.0 > latitude) && (latitude >= 24.0))
|
|
letterDesignator = 'R';
|
|
else if ((24.0 > latitude) && (latitude >= 16.0))
|
|
letterDesignator = 'Q';
|
|
else if ((16.0 > latitude) && (latitude >= 8.0))
|
|
letterDesignator = 'P';
|
|
else if ((8.0 > latitude) && (latitude >= 0.0))
|
|
letterDesignator = 'N';
|
|
else if ((0.0 > latitude) && (latitude >= -8.0))
|
|
letterDesignator = 'M';
|
|
else if ((-8.0 > latitude) && (latitude >= -16.0))
|
|
letterDesignator = 'L';
|
|
else if ((-16.0 > latitude) && (latitude >= -24.0))
|
|
letterDesignator = 'K';
|
|
else if ((-24.0 > latitude) && (latitude >= -32.0))
|
|
letterDesignator = 'J';
|
|
else if ((-32.0 > latitude) && (latitude >= -40.0))
|
|
letterDesignator = 'H';
|
|
else if ((-40.0 > latitude) && (latitude >= -48.0))
|
|
letterDesignator = 'G';
|
|
else if ((-48.0 > latitude) && (latitude >= -56.0))
|
|
letterDesignator = 'F';
|
|
else if ((-56.0 > latitude) && (latitude >= -64.0))
|
|
letterDesignator = 'E';
|
|
else if ((-64.0 > latitude) && (latitude >= -72.0))
|
|
letterDesignator = 'D';
|
|
else if ((-72.0 > latitude) && (latitude >= -80.0))
|
|
letterDesignator = 'C';
|
|
else
|
|
letterDesignator = 'Z'; // This is here as an error flag to show that the
|
|
// Latitude is outside the UTM limits
|
|
return letterDesignator;
|
|
}
|
|
|
|
} // namespace models
|
|
|
|
MYNTEYE_END_NAMESPACE
|