MRPT  2.0.1
test.cpp
Go to the documentation of this file.
1 /* +------------------------------------------------------------------------+
2  | Mobile Robot Programming Toolkit (MRPT) |
3  | https://www.mrpt.org/ |
4  | |
5  | Copyright (c) 2005-2020, Individual contributors, see AUTHORS file |
6  | See: https://www.mrpt.org/Authors - All rights reserved. |
7  | Released under BSD License. See: https://www.mrpt.org/License |
8  +------------------------------------------------------------------------+ */
9 
18 #include <mrpt/poses/CPose3D.h>
19 #include <mrpt/random.h>
20 #include <mrpt/system/CTicTac.h>
21 #include <iostream>
22 
23 using namespace mrpt;
24 using namespace mrpt::gui;
25 using namespace mrpt::math;
26 using namespace mrpt::random;
27 using namespace mrpt::poses;
28 using namespace mrpt::system;
29 using namespace std;
30 
32 
33 // ------------------------------------------------------
34 // TestRANSACPlanes
35 // ------------------------------------------------------
36 void TestRANSACPlanes()
37 {
39 
40  // Generate random points:
41  // ------------------------------------
42  const size_t N_PLANES = 3;
43 
44  const size_t N_plane = 300;
45  const size_t N_noise = 300;
46 
47  const double PLANE_EQ[N_PLANES][4] = {
48  {1, -1, 1, -2}, {1, +1.5, 1, -1}, {0, -1, 1, +2}};
49 
50  CVectorDouble xs, ys, zs;
51  for (size_t p = 0; p < N_PLANES; p++)
52  {
53  for (size_t i = 0; i < N_plane; i++)
54  {
55  const double xx =
56  getRandomGenerator().drawUniform(-3, 3) + 5 * cos(0.4 * p);
57  const double yy =
58  getRandomGenerator().drawUniform(-3, 3) + 5 * sin(0.4 * p);
59  const double zz =
60  -(PLANE_EQ[p][3] + PLANE_EQ[p][0] * xx + PLANE_EQ[p][1] * yy) /
61  PLANE_EQ[p][2];
62  xs.push_back(xx);
63  ys.push_back(yy);
64  zs.push_back(zz);
65  }
66  }
67 
68  for (size_t i = 0; i < N_noise; i++)
69  {
70  xs.push_back(getRandomGenerator().drawUniform(-7, 7));
71  ys.push_back(getRandomGenerator().drawUniform(-7, 7));
72  zs.push_back(getRandomGenerator().drawUniform(-7, 7));
73  }
74 
75  // Run RANSAC
76  // ------------------------------------
77  vector<pair<size_t, TPlane>> detectedPlanes;
78  const double DIST_THRESHOLD = 0.05;
79 
80  CTicTac tictac;
81 
82  ransac_detect_3D_planes(xs, ys, zs, detectedPlanes, DIST_THRESHOLD, 40);
83 
84  // Display output:
85  cout << "RANSAC method: ransac_detect_3D_planes" << endl;
86  cout << " Computation time: " << tictac.Tac() * 1000.0 << " ms" << endl;
87  cout << " " << detectedPlanes.size() << " planes detected." << endl;
88 
89  // Show GUI
90  // --------------------------
91  win = mrpt::gui::CDisplayWindow3D::Create("RANSAC: 3D planes", 500, 500);
92 
94 
95  scene->insert(opengl::CGridPlaneXY::Create(-20, 20, -20, 20, 0, 1));
96  scene->insert(opengl::stock_objects::CornerXYZ());
97 
98  for (vector<pair<size_t, TPlane>>::iterator p = detectedPlanes.begin();
99  p != detectedPlanes.end(); ++p)
100  {
101  auto glPlane = opengl::CTexturedPlane::Create(-10, 10, -10, 10);
102 
103  TPose3D glPlanePose;
104  p->second.getAsPose3D(glPlanePose);
105  glPlane->setPose(glPlanePose);
106 
107  glPlane->setColor(
108  getRandomGenerator().drawUniform<float>(0, 1),
109  getRandomGenerator().drawUniform<float>(0, 1),
110  getRandomGenerator().drawUniform<float>(0, 1), 0.6f);
111 
112  scene->insert(glPlane);
113  }
114 
115  {
116  auto points = opengl::CPointCloud::Create();
117  points->setColor(0, 0, 1);
118  points->setPointSize(3);
119  points->enableColorFromZ();
120 
121  // Convert double -> float:
122  vector<float> xsf, ysf, zsf;
126 
127  points->setAllPoints(xsf, ysf, zsf);
128 
129  scene->insert(points);
130  }
131 
132  win->get3DSceneAndLock() = scene;
133  win->unlockAccess3DScene();
134  win->forceRepaint();
135 
136  win->waitForKey();
137 }
138 
139 // ------------------------------------------------------
140 // TestRANSACLines
141 // ------------------------------------------------------
142 void TestRANSACLines()
143 {
145 
146  // Generate random points in 2D
147  // ------------------------------------
148  const size_t N_LINES = 4;
149 
150  const size_t N_line = 30;
151  const size_t N_noise = 50;
152 
153  const double LINE_EQ[N_LINES][3] = {
154  {1, -1, -2}, {1, +1.5, -1}, {0, -1, +2}, {0.5, -0.3, +1}};
155 
156  CVectorDouble xs, ys;
157  for (size_t p = 0; p < N_LINES; p++)
158  {
159  for (size_t i = 0; i < N_line; i++)
160  {
161  const double xx = getRandomGenerator().drawUniform(-10, 10);
162  const double yy =
164  (LINE_EQ[p][2] + LINE_EQ[p][0] * xx) / LINE_EQ[p][1];
165  xs.push_back(xx);
166  ys.push_back(yy);
167  }
168  }
169 
170  for (size_t i = 0; i < N_noise; i++)
171  {
172  xs.push_back(getRandomGenerator().drawUniform(-15, 15));
173  ys.push_back(getRandomGenerator().drawUniform(-15, 15));
174  }
175 
176  // Run RANSAC
177  // ------------------------------------
178  vector<pair<size_t, TLine2D>> detectedLines;
179  const double DIST_THRESHOLD = 0.2;
180 
181  CTicTac tictac;
182 
183  ransac_detect_2D_lines(xs, ys, detectedLines, DIST_THRESHOLD, 20);
184 
185  // Display output:
186  cout << "RANSAC method: ransac_detect_2D_lines" << endl;
187  cout << " Computation time: " << tictac.Tac() * 1000.0 << " ms" << endl;
188  cout << " " << detectedLines.size() << " lines detected." << endl;
189 
190  // Show GUI
191  // --------------------------
192  mrpt::gui::CDisplayWindowPlots win2("Set of points", 500, 500);
193 
194  win2.plot(xs, ys, ".b4", "points");
195 
196  unsigned int n = 0;
197  for (vector<pair<size_t, TLine2D>>::iterator p = detectedLines.begin();
198  p != detectedLines.end(); ++p)
199  {
200  CVectorDouble lx(2), ly(2);
201  lx[0] = -15;
202  lx[1] = 15;
203  for (CVectorDouble::Index q = 0; q < lx.size(); q++)
204  ly[q] = -(p->second.coefs[2] + p->second.coefs[0] * lx[q]) /
205  p->second.coefs[1];
206  win2.plot(lx, ly, "r-1", format("line_%u", n++));
207  }
208 
209  win2.axis_fit();
210  win2.axis_equal();
211 
212  win2.waitForKey();
213 }
214 
215 // ------------------------------------------------------
216 // MAIN
217 // ------------------------------------------------------
218 int main()
219 {
220  try
221  {
223  cout << endl << "Now running detection of lines..." << endl << endl;
224  TestRANSACLines();
225 
226  win.reset();
227 
228  return 0;
229  }
230  catch (const std::exception& e)
231  {
232  std::cerr << "MRPT error: " << mrpt::exception_to_str(e) << std::endl;
233  return -1;
234  }
235  catch (...)
236  {
237  printf("Untyped exception!!");
238  return -1;
239  }
240 }
A namespace of pseudo-random numbers generators of diferent distributions.
double Tac() noexcept
Stops the stopwatch.
Definition: CTicTac.cpp:86
static Ptr Create(Args &&... args)
Create a GUI window and display plots with MATLAB-like interfaces and commands.
std::string std::string format(std::string_view fmt, ARGS &&... args)
Definition: format.h:26
void TestRANSACPlanes()
void copy_container_typecasting(const src_container &src, dst_container &trg)
Copy all the elements in a container (vector, deque, list) into a different one performing the approp...
void randomize(const uint32_t seed)
Initialize the PRNG from the given random seed.
A high-performance stopwatch, with typical resolution of nanoseconds.
STL namespace.
static CDisplayWindow3D::Ptr Create(const std::string &windowCaption, unsigned int initialWindowWidth=400, unsigned int initialWindowHeight=300)
Class factory returning a smart pointer.
void push_back(const T &val)
static Ptr Create(Args &&... args)
Definition: CPointCloud.h:49
void ransac_detect_3D_planes(const CVectorDynamic< NUMTYPE > &x, const CVectorDynamic< NUMTYPE > &y, const CVectorDynamic< NUMTYPE > &z, std::vector< std::pair< size_t, TPlane >> &out_detected_planes, const double threshold, const size_t min_inliers_for_valid_plane=10)
Fit a number of 3-D planes to a given point cloud, automatically determining the number of existing p...
return_t drawUniform(const double Min, const double Max)
Generate a uniformly distributed pseudo-random number using the MT19937 algorithm, scaled to the selected range.
This base provides a set of functions for maths stuff.
return_t drawGaussian1D(const double mean, const double std)
Generate a normally distributed pseudo-random number.
mrpt::gui::CDisplayWindow3D::Ptr win
Classes for 2D/3D geometry representation, both of single values and probability density distribution...
void TestRANSACLines()
void ransac_detect_2D_lines(const CVectorDynamic< NUMTYPE > &x, const CVectorDynamic< NUMTYPE > &y, std::vector< std::pair< size_t, TLine2D >> &out_detected_lines, const double threshold, const size_t min_inliers_for_valid_line=5)
Fit a number of 2-D lines to a given point cloud, automatically determining the number of existing li...
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
CSetOfObjects::Ptr CornerXYZ(float scale=1.0)
Returns three arrows representing a X,Y,Z 3D corner.
Lightweight 3D pose (three spatial coordinates, plus three angular coordinates).
Definition: TPose3D.h:24
std::string exception_to_str(const std::exception &e)
Builds a nice textual representation of a nested exception, which if generated using MRPT macros (THR...
Definition: exceptions.cpp:59
static Ptr Create(Args &&... args)
Definition: COpenGLScene.h:58
Classes for creating GUI windows for 2D and 3D visualization.
Definition: about_box.h:14
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
static Ptr Create(Args &&... args)
Definition: CGridPlaneXY.h:31



Page generated by Doxygen 1.8.14 for MRPT 2.0.1 Git: 0fef1a6d7 Fri Apr 3 23:00:21 2020 +0200 at vie abr 3 23:20:28 CEST 2020