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 
10 /**
11  * icp3d
12  * Execute an Iterative Closest Point algorithm using two 3D point clouds.
13  */
14 
15 #include <mrpt/slam/CICP.h>
16 
21 #include <mrpt/opengl/CDisk.h>
23 #include <mrpt/opengl/CSphere.h>
25 #include <mrpt/poses/CPose3DPDF.h>
26 #include <iostream>
27 
28 using namespace std;
29 using namespace mrpt;
30 using namespace mrpt::gui;
31 using namespace mrpt::opengl;
32 using namespace mrpt::poses;
33 using namespace mrpt::slam;
34 using namespace mrpt::maps;
35 using namespace mrpt::obs;
36 
37 // Increase this values to get more precision. It will also increase run time.
38 const size_t HOW_MANY_YAWS = 120;
39 const size_t HOW_MANY_PITCHS = 120;
40 
41 // The scans of the 3D object, taken from 2 different places:
42 vector<CObservation2DRangeScan> sequence_scans1, sequence_scans2;
43 
44 // The two origins for the 3D scans
45 CPose3D viewpoint1(-0.3, 0.7, 3, 5.0_deg, 80.0_deg, 3.0_deg);
46 CPose3D viewpoint2(0.5, -0.2, 2.6, -5.0_deg, 100.0_deg, -7.0_deg);
47 
48 CPose3D SCAN2_POSE_ERROR(0.15, -0.07, 0.10, -0.03, 0.1, 0.1);
49 
50 /**
51  * Generate 3 objects to work with - 1 sphere, 2 disks
52  */
54 {
55  CSphere::Ptr sph = CSphere::Create(0.5);
56  sph->setLocation(0, 0, 0);
57  sph->setColor(1, 0, 0);
58  world->insert(sph);
59 
60  CDisk::Ptr pln = opengl::CDisk::Create();
61  pln->setDiskRadius(2);
62  pln->setPose(CPose3D(0, 0, 0, 0, 5.0_deg, 5.0_deg));
63  pln->setColor(0.8, 0, 0);
64  world->insert(pln);
65 
66  {
67  CDisk::Ptr pln = opengl::CDisk::Create();
68  pln->setDiskRadius(2);
69  pln->setPose(CPose3D(0, 0, 0, 30.0_deg, -20.0_deg, -2.0_deg));
70  pln->setColor(0.9, 0, 0);
71  world->insert(pln);
72  }
73 }
74 
75 void test_icp3D()
76 {
77  // Create the reference objects:
78  COpenGLScene::Ptr scene1 = COpenGLScene::Create();
79  COpenGLScene::Ptr scene2 = COpenGLScene::Create();
80  COpenGLScene::Ptr scene3 = COpenGLScene::Create();
81 
83  CGridPlaneXY::Create(-20, 20, -20, 20, 0, 1);
84  plane1->setColor(0.3f, 0.3f, 0.3f);
85  scene1->insert(plane1);
86  scene2->insert(plane1);
87  scene3->insert(plane1);
88 
89  CSetOfObjects::Ptr world = CSetOfObjects::Create();
90  generateObjects(world);
91  scene1->insert(world);
92 
93  // Perform the 3D scans:
94  CAngularObservationMesh::Ptr aom1 = CAngularObservationMesh::Create();
95  CAngularObservationMesh::Ptr aom2 = CAngularObservationMesh::Create();
96 
97  cout << "Performing ray-tracing..." << endl;
98  CAngularObservationMesh::trace2DSetOfRays(
99  scene1, viewpoint1, aom1,
100  CAngularObservationMesh::TDoubleRange::CreateFromAperture(
102  CAngularObservationMesh::TDoubleRange::CreateFromAperture(
103  M_PI, HOW_MANY_YAWS));
104  CAngularObservationMesh::trace2DSetOfRays(
105  scene1, viewpoint2, aom2,
106  CAngularObservationMesh::TDoubleRange::CreateFromAperture(
108  CAngularObservationMesh::TDoubleRange::CreateFromAperture(
109  M_PI, HOW_MANY_YAWS));
110  cout << "Ray-tracing done" << endl;
111 
112  // Put the viewpoints origins:
113  {
115  origin1->setPose(viewpoint1);
116  origin1->setScale(0.6);
117  scene1->insert(origin1);
118  scene2->insert(origin1);
119  }
120  {
122  origin2->setPose(viewpoint2);
123  origin2->setScale(0.6);
124  scene1->insert(origin2);
125  scene2->insert(origin2);
126  }
127 
128  // Show the scanned points:
129  CSimplePointsMap M1, M2;
130 
131  aom1->generatePointCloud(&M1);
132  aom2->generatePointCloud(&M2);
133 
134  // Create the wrongly-localized M2:
135  CSimplePointsMap M2_noisy;
136  M2_noisy = M2;
138 
139  CSetOfObjects::Ptr PTNS1 = CSetOfObjects::Create();
140  CSetOfObjects::Ptr PTNS2 = CSetOfObjects::Create();
141 
142  M1.renderOptions.color = mrpt::img::TColorf(1, 0, 0);
143  M1.getAs3DObject(PTNS1);
144 
145  M2_noisy.renderOptions.color = mrpt::img::TColorf(0, 0, 1);
146  M2_noisy.getAs3DObject(PTNS2);
147 
148  scene2->insert(PTNS1);
149  scene2->insert(PTNS2);
150 
151  // --------------------------------------
152  // Do the ICP-3D
153  // --------------------------------------
154  CICP icp;
155  CICP::TReturnInfo icp_info;
156 
157  icp.options.thresholdDist = 0.40;
158  icp.options.thresholdAng = 0;
159 
160  std::vector<double> xs, ys, zs;
161  M1.getAllPoints(xs, ys, ys);
162  cout << "Size of xs in M1: " << xs.size() << endl;
163  M2.getAllPoints(xs, ys, ys);
164  cout << "Size of xs in M2: " << xs.size() << endl;
165 
166  CPose3DPDF::Ptr pdf = icp.Align3D(
167  &M2_noisy, // Map to align
168  &M1, // Reference map
169  CPose3D(), // Initial gross estimate
170  icp_info);
171 
172  CPose3D mean = pdf->getMeanVal();
173 
174  cout << "ICP run took " << icp_info.executionTime << " secs." << endl;
175  cout << "Goodness: " << 100 * icp_info.goodness
176  << "% , # of iterations= " << icp_info.nIterations
177  << " Quality: " << icp_info.quality << endl;
178  cout << "ICP output: mean= " << mean << endl;
179  cout << "Real displacement: " << SCAN2_POSE_ERROR << endl;
180 
181  // Aligned maps:
182  CSetOfObjects::Ptr PTNS2_ALIGN = CSetOfObjects::Create();
183 
185  M2_noisy.getAs3DObject(PTNS2_ALIGN);
186 
187  scene3->insert(PTNS1);
188  scene3->insert(PTNS2_ALIGN);
189 
190  // Show in Windows:
191  CDisplayWindow3D window("ICP-3D demo: scene", 500, 500);
192  CDisplayWindow3D window2("ICP-3D demo: UNALIGNED scans", 500, 500);
193  CDisplayWindow3D window3("ICP-3D demo: ICP-ALIGNED scans", 500, 500);
194 
195  window.setPos(10, 10);
196  window2.setPos(530, 10);
197  window3.setPos(10, 520);
198 
199  window.get3DSceneAndLock() = scene1;
200  window.unlockAccess3DScene();
201 
202  window2.get3DSceneAndLock() = scene2;
203  window2.unlockAccess3DScene();
204 
205  window3.get3DSceneAndLock() = scene3;
206  window3.unlockAccess3DScene();
207 
208  std::this_thread::sleep_for(20ms);
209  window.forceRepaint();
210  window2.forceRepaint();
211 
212  window.setCameraElevationDeg(15);
213  window.setCameraAzimuthDeg(90);
214  window.setCameraZoom(15);
215 
216  window2.setCameraElevationDeg(15);
217  window2.setCameraAzimuthDeg(90);
218  window2.setCameraZoom(15);
219 
220  window3.setCameraElevationDeg(15);
221  window3.setCameraAzimuthDeg(90);
222  window3.setCameraZoom(15);
223 
224  cout << "Press any key to exit..." << endl;
225  window.waitForKey();
226 }
227 
228 int main()
229 {
230  try
231  {
232  test_icp3D();
233  return 0;
234  }
235  catch (exception& e)
236  {
237  cout << "Error: " << e.what() << '.' << endl;
238  return -1;
239  }
240  catch (...)
241  {
242  cout << "Unknown Error.\n";
243  return -1;
244  }
245 }
void changeCoordinatesReference(const mrpt::poses::CPose2D &b)
Replace each point by (pose compounding operator).
Definition: CPointsMap.cpp:545
mrpt::img::TColorf color
Color of points.
Definition: CPointsMap.h:327
Several implementations of ICP (Iterative closest point) algorithms for aligning two point maps or a ...
Definition: CICP.h:64
void getAs3DObject(mrpt::opengl::CSetOfObjects::Ptr &outObj) const override
Returns a 3D object representing the map.
Definition: CPointsMap.cpp:766
A cloud of points in 2D or 3D, which can be built from a sequence of laser scans. ...
unsigned int nIterations
The number of executed iterations until convergence.
Definition: CICP.h:196
STL namespace.
void generateObjects(CSetOfObjects::Ptr &world)
TConfigParams options
The options employed by the ICP align.
Definition: CICP.h:180
void getAllPoints(VECTOR &xs, VECTOR &ys, VECTOR &zs, size_t decimation=1) const
Returns a copy of the 2D/3D points as a std::vector of float coordinates.
Definition: CPointsMap.h:571
This namespace contains representation of robot actions and observations.
CPose3D viewpoint1(-0.3, 0.7, 3, 5.0_deg, 80.0_deg, 3.0_deg)
double goodness
A goodness measure for the alignment, it is a [0,1] range indicator of percentage of correspondences...
Definition: CICP.h:200
vector< CObservation2DRangeScan > sequence_scans2
Classes for 2D/3D geometry representation, both of single values and probability density distribution...
const size_t HOW_MANY_YAWS
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
TRenderOptions renderOptions
Definition: CPointsMap.h:331
A class used to store a 3D pose (a 3D translation + a rotation in 3D).
Definition: CPose3D.h:85
CSetOfObjects::Ptr CornerXYZ(float scale=1.0)
Returns three arrows representing a X,Y,Z 3D corner.
mrpt::poses::CPose3DPDF::Ptr Align3D(const mrpt::maps::CMetricMap *m1, const mrpt::maps::CMetricMap *m2, const mrpt::poses::CPose3D &grossEst, mrpt::optional_ref< TMetricMapAlignmentResult > outInfo=std::nullopt)
CPose3D SCAN2_POSE_ERROR(0.15, -0.07, 0.10, -0.03, 0.1, 0.1)
An RGBA color - floats in the range [0,1].
Definition: TColor.h:88
void test_icp3D()
The ICP algorithm return information.
Definition: CICP.h:190
The namespace for 3D scene representation and rendering.
Definition: CGlCanvasBase.h:13
double mean(const CONTAINER &v)
Computes the mean value of a vector.
Classes for creating GUI windows for 2D and 3D visualization.
Definition: about_box.h:14
CPose3D viewpoint2(0.5, -0.2, 2.6, -5.0_deg, 100.0_deg, -7.0_deg)
double thresholdDist
Initial threshold distance for two points to become a correspondence.
Definition: CICP.h:114
const size_t HOW_MANY_PITCHS
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.
vector< CObservation2DRangeScan > sequence_scans1
double quality
A measure of the &#39;quality&#39; of the local minimum of the sqr.
Definition: CICP.h:205



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