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 #include <mrpt/core/round.h>
13 #include <mrpt/opengl/CSphere.h>
14 #include <mrpt/random.h>
15 #include <mrpt/system/CTicTac.h>
16 #include <mrpt/system/os.h>
17 #include <iostream>
18 
19 using namespace mrpt;
20 using namespace std;
21 using namespace mrpt::gui;
22 using namespace mrpt::random;
23 using namespace mrpt::opengl;
24 
25 const size_t N_MASSES = 750;
26 
27 const double BOX = 500;
28 const double V0 = 100;
29 const double MASS_MIN = log(40.0), MASS_MAX = log(100.0);
30 const double M2R = 2.0;
31 const double LARGEST_STEP = 0.001;
32 const double G = 300;
33 const double COLLIS_LOSS = 0.98;
34 
35 struct TMass
36 {
37  TMass() : obj3d() {}
38  double x{0}, y{0}, z{0};
39  double vx{0}, vy{0}, vz{0};
40  double mass{1};
41  double radius;
43 };
44 
45 void simulateGravity(vector<TMass>& objs, double At);
46 
47 // ------------------------------------------------------
48 // GravityDemo
49 // ------------------------------------------------------
50 void GravityDemo()
51 {
53  "MRPT example: 3D gravity simulator- JLBC 2008", 1000, 700);
54 
56 
57  win.setCameraElevationDeg(50.0f);
58  win.setCameraZoom(1000);
59 
60  COpenGLScene::Ptr& theScene = win.get3DSceneAndLock();
61 
62  // Modify the scene:
63  // ------------------------------------------------------
64  {
66  opengl::CGridPlaneXY::Create(-2000, 2000, -2000, 2000, 0, 100);
67  obj->setColor(0.3f, 0.3f, 0.3f);
68  theScene->insert(obj);
69  }
70 
71  // Create the masses:
72  // ----------------------------------------------------
73 
74  vector<TMass> masses(N_MASSES);
75 
76  // Init at random poses & create opengl objects:
77  for (size_t i = 0; i < N_MASSES; i++)
78  {
79  masses[i].x = getRandomGenerator().drawUniform(-BOX, BOX);
80  masses[i].y = getRandomGenerator().drawUniform(-BOX, BOX);
81  masses[i].z = getRandomGenerator().drawUniform(-BOX, BOX) / 10;
82 
83  double a = atan2(masses[i].y, masses[i].x);
84 
85  masses[i].vx = -V0 * sin(a) +
86  getRandomGenerator().drawUniform(-V0 * 0.01, V0 * 0.01);
87  masses[i].vy = V0 * cos(a) +
88  getRandomGenerator().drawUniform(-V0 * 0.01, V0 * 0.01);
89  masses[i].vz = 0; // getRandomGenerator().drawUniform(-V0,V0);
90 
91  masses[i].mass =
92  exp(getRandomGenerator().drawUniform(MASS_MIN, MASS_MAX));
93  opengl::CSphere::Ptr& obj = masses[i].obj3d = opengl::CSphere::Create();
94 
95  obj->setColor(
96  getRandomGenerator().drawUniform<float>(0.1, 0.9),
97  getRandomGenerator().drawUniform<float>(0.1, 0.9),
98  getRandomGenerator().drawUniform<float>(0.1, 0.9));
99 
100  masses[i].radius = M2R * pow(masses[i].mass, 1.0 / 3.0);
101  obj->setRadius(mrpt::d2f(masses[i].radius)); // Guess why ^(1/3) ;-)
102 
103  obj->setLocation(masses[i].x, masses[i].y, masses[i].z);
104  theScene->insert(obj);
105  }
106 
107  // IMPORTANT!!! IF NOT UNLOCKED, THE WINDOW WILL NOT BE UPDATED!
108  win.unlockAccess3DScene();
109 
110  mrpt::system::CTicTac tictac;
111  tictac.Tic();
112 
113  double t0 = tictac.Tac();
114 
115  while (!mrpt::system::os::kbhit() && win.isOpen())
116  {
117  // Move the scene:
118  double t1 = tictac.Tac();
119  double At = t1 - t0;
120  t0 = t1;
121 
122  // Simulate a At, possibly in several small steps:
123  // Update the 3D scene:
124  win.get3DSceneAndLock();
125 
126  size_t n_steps = mrpt::round(ceil(At / LARGEST_STEP) + 1);
127  double At_steps = At / n_steps;
128  n_steps = min(n_steps, size_t(3));
129  for (size_t j = 0; j < n_steps; j++) simulateGravity(masses, At_steps);
130 
131  for (size_t i = 0; i < masses.size(); i++)
132  {
133  opengl::CSphere::Ptr& obj = masses[i].obj3d;
134  obj->setLocation(masses[i].x, masses[i].y, masses[i].z);
135  }
136  // IMPORTANT!!! IF NOT UNLOCKED, THE WINDOW WILL NOT BE UPDATED!
137  win.unlockAccess3DScene();
138 
139  // Update window:
140  win.forceRepaint();
141  std::this_thread::sleep_for(1ms);
142  };
143 }
144 
145 struct TForce
146 {
147  TForce() { f[0] = f[1] = f[2] = 0; }
148  double f[3];
149 };
150 
151 void simulateGravity(vector<TMass>& objs, double At)
152 {
153  const size_t N = objs.size();
154 
155  vector<TForce> forces(N);
156 
157  // Index in the array must be larger than its content!!
158  vector<pair<size_t, double>> lstMass_i_joins_j(
159  N, pair<size_t, double>(string::npos, 10000.0));
160 
161  for (size_t i = 0; i < (N - 1); i++)
162  {
163  const double Ri = objs[i].radius;
164 
165  // Compute overall gravity force:
166  for (size_t j = i + 1; j < N; j++)
167  {
168  double Ax = objs[j].x - objs[i].x;
169  double Ay = objs[j].y - objs[i].y;
170  double Az = objs[j].z - objs[i].z;
171  double D2 = square(Ax) + square(Ay) + square(Az);
172 
173  double D = sqrt(D2);
174  if (D == 0) continue;
175 
176  const double Rj = objs[j].radius;
177 
178  if (D < (Ri + Rj)) // Collission!!
179  {
180  // Index in the array must be larger than its content!!
181  if (D < lstMass_i_joins_j[j].second)
182  {
183  lstMass_i_joins_j[j].first = i;
184  lstMass_i_joins_j[j].second = D;
185  }
186  }
187  else
188  {
189  double K =
190  G * objs[i].mass * objs[j].mass / square(max(D, 1.0));
191  double D_1 = 1.0 / D;
192  Ax *= D_1;
193  Ay *= D_1;
194  Az *= D_1;
195 
196  const double fx = Ax * K;
197  const double fy = Ay * K;
198  const double fz = Az * K;
199 
200  forces[i].f[0] += fx;
201  forces[i].f[1] += fy;
202  forces[i].f[2] += fz;
203 
204  forces[j].f[0] -= fx;
205  forces[j].f[1] -= fy;
206  forces[j].f[2] -= fz;
207  }
208  }
209  }
210 
211  // F = m a
212  for (size_t i = 0; i < N; i++)
213  {
214  const double M_1 = 1.0 / objs[i].mass;
215 
216  forces[i].f[0] *= M_1;
217  forces[i].f[1] *= M_1;
218  forces[i].f[2] *= M_1;
219 
220  objs[i].vx += forces[i].f[0] * At;
221  objs[i].vy += forces[i].f[1] * At;
222  objs[i].vz += forces[i].f[2] * At;
223 
224  objs[i].x += objs[i].vx * At;
225  objs[i].y += objs[i].vy * At;
226  objs[i].z += objs[i].vz * At;
227  }
228 
229  // return;
230  // Join masses that collided:
231  for (int i = N - 1; i >= 0; i--)
232  {
233  const size_t newObj = lstMass_i_joins_j[i].first;
234  if (newObj == string::npos) continue;
235 
236  const double Mi = objs[i].mass;
237  const double Mj = objs[newObj].mass;
238  const double newMass = Mi + Mj;
239  const double newMass_1 = 1.0 / newMass;
240 
241  objs[newObj].vx =
242  COLLIS_LOSS * newMass_1 * (Mj * objs[newObj].vx + Mi * objs[i].vx);
243  objs[newObj].vy =
244  COLLIS_LOSS * newMass_1 * (Mj * objs[newObj].vy + Mi * objs[i].vy);
245  objs[newObj].vz =
246  COLLIS_LOSS * newMass_1 * (Mj * objs[newObj].vz + Mi * objs[i].vz);
247 
248  objs[newObj].x = newMass_1 * (Mj * objs[newObj].x + Mi * objs[i].x);
249  objs[newObj].y = newMass_1 * (Mj * objs[newObj].y + Mi * objs[i].y);
250  objs[newObj].z = newMass_1 * (Mj * objs[newObj].z + Mi * objs[i].z);
251 
252  objs[newObj].mass = newMass;
253  objs[newObj].radius = M2R * pow(newMass, 1.0 / 3.0);
254  objs[newObj].obj3d->setRadius(mrpt::d2f(objs[newObj].radius));
255 
256  objs[i].obj3d->setVisibility(false); // Hide sphere
257  objs.erase(objs.begin() + i);
258  }
259 }
260 
261 // ------------------------------------------------------
262 // MAIN
263 // ------------------------------------------------------
264 int main()
265 {
266  try
267  {
268  GravityDemo();
269  return 0;
270  }
271  catch (const std::exception& e)
272  {
273  std::cerr << "MRPT error: " << mrpt::exception_to_str(e) << std::endl;
274  return -1;
275  }
276  catch (...)
277  {
278  printf("Untyped exception!!");
279  return -1;
280  }
281 }
A namespace of pseudo-random numbers generators of diferent distributions.
const double MASS_MAX
double Tac() noexcept
Stops the stopwatch.
Definition: CTicTac.cpp:86
const double LARGEST_STEP
void GravityDemo()
const double G
const double V0
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.
float d2f(const double d)
shortcut for static_cast<float>(double)
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.
mrpt::gui::CDisplayWindow3D::Ptr win
const size_t N_MASSES
return_t square(const num_t x)
Inline function for the square of a number.
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
The namespace for 3D scene representation and rendering.
Definition: CGlCanvasBase.h:13
bool kbhit() noexcept
An OS-independent version of kbhit, which returns true if a key has been pushed.
Definition: os.cpp:392
static Ptr Create(Args &&... args)
Definition: CSphere.h:31
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
void simulateGravity(vector< TMass > &objs, double At)
const double COLLIS_LOSS
Classes for creating GUI windows for 2D and 3D visualization.
Definition: about_box.h:14
void Tic() noexcept
Starts the stopwatch.
Definition: CTicTac.cpp:75
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
static Ptr Create(Args &&... args)
Definition: CGridPlaneXY.h:31
const double M2R
const double MASS_MIN
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.
const double BOX
int round(const T value)
Returns the closer integer (int) to x.
Definition: round.h:24



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