27 const double BOX = 500;
28 const double V0 = 100;
30 const double M2R = 2.0;
38 double x{0}, y{0}, z{0};
39 double vx{0}, vy{0}, vz{0};
53 "MRPT example: 3D gravity simulator- JLBC 2008", 1000, 700);
57 win.setCameraElevationDeg(50.0f);
58 win.setCameraZoom(1000);
67 obj->setColor(0.3f, 0.3f, 0.3f);
68 theScene->insert(obj);
77 for (
size_t i = 0; i <
N_MASSES; i++)
83 double a = atan2(masses[i].y, masses[i].x);
85 masses[i].vx = -
V0 * sin(a) +
87 masses[i].vy =
V0 * cos(a) +
100 masses[i].radius =
M2R * pow(masses[i].mass, 1.0 / 3.0);
101 obj->setRadius(
mrpt::d2f(masses[i].radius));
103 obj->setLocation(masses[i].x, masses[i].y, masses[i].z);
104 theScene->insert(obj);
108 win.unlockAccess3DScene();
113 double t0 = tictac.
Tac();
118 double t1 = tictac.
Tac();
124 win.get3DSceneAndLock();
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);
131 for (
size_t i = 0; i < masses.size(); i++)
134 obj->setLocation(masses[i].x, masses[i].y, masses[i].z);
137 win.unlockAccess3DScene();
141 std::this_thread::sleep_for(1ms);
147 TForce() { f[0] = f[1] = f[2] = 0; }
153 const size_t N = objs.size();
155 vector<TForce> forces(N);
158 vector<pair<size_t, double>> lstMass_i_joins_j(
159 N, pair<size_t, double>(string::npos, 10000.0));
161 for (
size_t i = 0; i < (N - 1); i++)
163 const double Ri = objs[i].radius;
166 for (
size_t j = i + 1; j < N; j++)
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;
174 if (D == 0)
continue;
176 const double Rj = objs[j].radius;
181 if (D < lstMass_i_joins_j[j].second)
183 lstMass_i_joins_j[j].first = i;
184 lstMass_i_joins_j[j].second = D;
190 G * objs[i].mass * objs[j].mass /
square(max(D, 1.0));
191 double D_1 = 1.0 / D;
196 const double fx = Ax * K;
197 const double fy = Ay * K;
198 const double fz = Az * K;
200 forces[i].f[0] += fx;
201 forces[i].f[1] += fy;
202 forces[i].f[2] += fz;
204 forces[j].f[0] -= fx;
205 forces[j].f[1] -= fy;
206 forces[j].f[2] -= fz;
212 for (
size_t i = 0; i < N; i++)
214 const double M_1 = 1.0 / objs[i].mass;
216 forces[i].f[0] *= M_1;
217 forces[i].f[1] *= M_1;
218 forces[i].f[2] *= M_1;
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;
224 objs[i].x += objs[i].vx * At;
225 objs[i].y += objs[i].vy * At;
226 objs[i].z += objs[i].vz * At;
231 for (
int i = N - 1; i >= 0; i--)
233 const size_t newObj = lstMass_i_joins_j[i].first;
234 if (newObj == string::npos)
continue;
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;
242 COLLIS_LOSS * newMass_1 * (Mj * objs[newObj].vx + Mi * objs[i].vx);
244 COLLIS_LOSS * newMass_1 * (Mj * objs[newObj].vy + Mi * objs[i].vy);
246 COLLIS_LOSS * newMass_1 * (Mj * objs[newObj].vz + Mi * objs[i].vz);
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);
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));
256 objs[i].obj3d->setVisibility(
false);
257 objs.erase(objs.begin() + i);
271 catch (
const std::exception& e)
278 printf(
"Untyped exception!!");
A namespace of pseudo-random numbers generators of diferent distributions.
double Tac() noexcept
Stops the stopwatch.
const double LARGEST_STEP
void randomize(const uint32_t seed)
Initialize the PRNG from the given random seed.
A high-performance stopwatch, with typical resolution of nanoseconds.
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
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.
bool kbhit() noexcept
An OS-independent version of kbhit, which returns true if a key has been pushed.
static Ptr Create(Args &&... args)
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...
void simulateGravity(vector< TMass > &objs, double At)
Classes for creating GUI windows for 2D and 3D visualization.
void Tic() noexcept
Starts the stopwatch.
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
static Ptr Create(Args &&... args)
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.
int round(const T value)
Returns the closer integer (int) to x.