reactphysics3d/documentation/UserManual/ReactPhysics3D-UserManual.tex

1926 lines
95 KiB
TeX
Raw Normal View History

2013-03-18 21:14:43 +00:00
\documentclass[a4paper,12pt]{article}
2016-04-09 20:52:25 +00:00
2013-03-18 21:14:43 +00:00
% packages
\usepackage[T1]{fontenc}
\usepackage[latin1]{inputenc}
\usepackage{amsmath}
\usepackage{amsfonts}
\usepackage{amssymb}
\usepackage{graphicx}
\usepackage{listings}
\usepackage{url}
2018-03-19 22:07:09 +00:00
\usepackage{hyperref}
2013-09-24 22:02:09 +00:00
\usepackage[usenames]{xcolor}
2018-03-19 22:07:09 +00:00
\hypersetup{
colorlinks,
linkcolor={blue!70!black},
citecolor={blue!200!black},
urlcolor={blue!80!black}
}
2013-03-18 21:14:43 +00:00
\usepackage[top=3cm, bottom=3cm, left=3cm, right=3cm]{geometry}
2018-03-19 22:07:09 +00:00
\usepackage[scaled]{helvet}
\renewcommand\familydefault{\sfdefault}
2013-09-24 22:02:09 +00:00
% Path to the pictures
\graphicspath{ {images/} }
% Customized style to display c++ source code
\lstdefinestyle{customcpp}{
backgroundcolor=\color{gray!15},
belowcaptionskip=1\baselineskip,
breaklines=true,
2018-03-19 22:07:09 +00:00
frame=single,
2013-09-24 22:02:09 +00:00
xleftmargin=\parindent,
language=C++,
showstringspaces=false,
2018-03-19 22:07:09 +00:00
basicstyle=\ttfamily\footnotesize,
2013-09-24 22:02:09 +00:00
keywordstyle=\bfseries\color{green!40!black},
commentstyle=\itshape\color{purple!40!black},
identifierstyle=\color{blue},
stringstyle=\color{orange},
}
\lstset{style=customcpp}
2013-03-18 21:14:43 +00:00
\input{title}
\begin{document}
\author{Daniel Chappuis}
\title{ReactPhysics3D library \\ User Manual}
\maketitle
\tableofcontents
\newpage
2016-04-09 20:52:25 +00:00
2013-03-18 21:14:43 +00:00
\section{Introduction}
ReactPhysics3D is an open source C++ physics engine library that can be used
in 3D simulations and games. The library is released under the ZLib license.
\section{Features}
2015-03-01 21:37:22 +00:00
The ReactPhysics3D library has the following features:
2013-03-18 21:14:43 +00:00
\begin{itemize}
2016-04-09 20:52:25 +00:00
\item Rigid body dynamics
\item Discrete collision detection
2018-03-06 18:29:16 +00:00
\item Collision shapes (Sphere, Box, Capsule, Convex Mesh, Static Concave Mesh, Height Field)
2015-03-01 21:37:22 +00:00
\item Multiple collision shapes per body
2016-04-09 20:52:25 +00:00
\item Broadphase collision detection (Dynamic AABB tree)
2018-03-06 18:29:16 +00:00
\item Narrowphase collision detection (SAT/GJK)
2013-03-18 21:14:43 +00:00
\item Collision response and friction (Sequential Impulses Solver)
2013-09-24 22:02:09 +00:00
\item Joints (Ball and Socket, Hinge, Slider, Fixed)
2015-03-01 21:37:22 +00:00
\item Collision filtering with categories
\item Ray casting
2013-09-24 22:02:09 +00:00
\item Sleeping technique for inactive bodies
2013-03-18 21:14:43 +00:00
\item Multi-platform (Windows, Linux, Mac OS X)
2018-03-06 18:29:16 +00:00
\item No external libraries (do not use STL containers)
2018-03-19 22:07:09 +00:00
\item Documentation (user manual and Doxygen API)
2016-04-09 20:52:25 +00:00
\item Testbed application with demos
2018-03-06 18:29:16 +00:00
\item Integrated Profiler
\item Logs
2013-03-18 21:14:43 +00:00
\item Unit tests
\end{itemize}
\section{License}
2016-04-09 20:52:25 +00:00
2013-09-24 22:02:09 +00:00
The ReactPhysics3D library is released under the open-source ZLib license. For more information, read the "LICENSE" file.
2013-03-18 21:14:43 +00:00
2013-09-24 22:02:09 +00:00
\section{Building the library}
2016-04-09 20:52:25 +00:00
\label{sec:building}
2013-03-18 21:14:43 +00:00
2016-04-09 20:52:25 +00:00
You should use the CMake software to generate the makefiles or the
2013-03-18 21:14:43 +00:00
project files for your IDE. CMake can be downloaded at
2015-03-01 21:37:22 +00:00
\url{http://www.cmake.org} or using your package-management program
2013-03-18 21:14:43 +00:00
(apt, yum, \dots) on Linux. Then, you will be able to compile the library to create the static library
2013-09-24 22:02:09 +00:00
file. In order to use ReactPhysics3D in your application, you can link your program with this static library.
2013-10-06 16:46:32 +00:00
If you have never used cmake before, you should read the page \url{http://www.cmake.org/cmake/help/runningcmake.html} as
2015-03-01 21:37:22 +00:00
it contains a lot of useful information. \\
2013-10-06 16:46:32 +00:00
2018-03-19 22:07:09 +00:00
It is also possible to compile the testbed application using CMake. This application contains different
physics demo scenes.
2016-04-09 20:52:25 +00:00
\subsection{CMake using the command line (Linux and Mac OS X)}
2013-03-18 21:14:43 +00:00
2015-03-01 21:37:22 +00:00
Now, we will see how to build the ReactPhysics3D library using the CMake tool with the command line.
First, create a folder where you want to build the library. Then go into that folder and run
the \texttt{ccmake} command: \\
2013-03-18 21:14:43 +00:00
2013-09-24 22:02:09 +00:00
\texttt{ccmake \textless path\_to\_library\_source\textgreater} \\
2013-03-18 21:14:43 +00:00
2015-03-01 21:37:22 +00:00
\begin{sloppypar}
2013-09-24 22:02:09 +00:00
where \texttt{\textless path\_to\_library\_source\textgreater} must be replaced
2018-03-06 18:29:16 +00:00
by the path to the \texttt{reactphysics3d-0.7.0/} folder. It is the folder that
2013-09-24 22:02:09 +00:00
contains the \texttt{CMakeLists.txt} file. Running this command will launch the CMake command line interface.
Hit the 'c' key to configure the project. There, you can also change some predefined variables (see section \ref{sec:cmakevariables} for more details)
and then, hit the 'c' key again. Once you have set all the values as you like, you can hit the 'g' key to generate the makefiles in the build directory
that you have created before and exit. \\
2013-03-18 21:14:43 +00:00
2018-03-19 22:07:09 +00:00
Now that you have generated the makefiles, you can compile the code to build the static library in the
2015-03-01 21:37:22 +00:00
\texttt{/lib} folder with the following command in your build directory: \\
\end{sloppypar}
2016-04-09 20:52:25 +00:00
2013-09-24 22:02:09 +00:00
\texttt{make}
2013-03-18 21:14:43 +00:00
2018-03-19 22:07:09 +00:00
TODO : Add "make install" information here
2013-09-24 22:02:09 +00:00
\subsection{CMake using the graphical interface (Linux, Mac OS X and Windows)}
2013-03-18 21:14:43 +00:00
2016-04-09 20:52:25 +00:00
You can also use the graphical user interface of CMake. To do this,
2015-03-01 21:37:22 +00:00
run the \texttt{cmake-gui} program. The program will ask you for the
2018-03-06 18:29:16 +00:00
source folder which is the \texttt{reactphysics3d-0.7.0/} folder of
2013-03-18 21:14:43 +00:00
the library. You will also have to select a folder where you want to
2016-04-09 20:52:25 +00:00
build the library and the testbed application. Select any empty folder that
2013-09-24 22:02:09 +00:00
is on your system. Then, you can click on \texttt{Configure}. CMake will ask you to choose an IDE that is on
2015-03-01 21:37:22 +00:00
your system. For instance, you can select Visual Studio, Qt Creator, XCode, ... Then, you
2013-09-24 22:02:09 +00:00
can change the compilation options. See section \ref{sec:cmakevariables} to see what are the possible options.
2018-03-19 22:07:09 +00:00
Once this is done, click on \texttt{Configure} again and finally on \texttt{Generate}. \\
2016-04-09 20:52:25 +00:00
2013-03-18 21:14:43 +00:00
Now, if you go into the folder you have chosen to build the
2013-09-24 22:02:09 +00:00
library, you should be able to open the project file that corresponds to your IDE and compile
2013-10-06 16:46:32 +00:00
the library. \\
2013-09-24 22:02:09 +00:00
\subsection{CMake Variables}
\label{sec:cmakevariables}
2015-03-01 21:37:22 +00:00
You can find bellow the different CMake variables that you can set before generating the makefiles:
2013-09-24 22:02:09 +00:00
\begin{description}
2013-10-06 16:46:32 +00:00
\item[CMAKE\_BUILD\_TYPE] If this variable is set to \texttt{Debug}, the library will be compiled in debugging mode.
2013-09-24 22:02:09 +00:00
This mode should be used during development stage to know where things might crash.
2015-03-01 21:37:22 +00:00
In debugging mode, the library might run a bit slow due to all the debugging information.
However, if this variable is set to \texttt{Release}, no debugging information is stored
and therefore, it will run much faster. This mode must be used when you compile the final
2018-03-26 20:10:39 +00:00
release of your application.
2016-04-09 20:52:25 +00:00
\item[COMPILE\_TESTBED] If this variable is \texttt{ON}, the tesbed application of the library will be compiled.
The testbed application uses OpenGL for rendering.
Take a look at the section \ref{sec:testbed} for more information about the testbed application.
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
\item[COMPILE\_TESTS] If this variable is \texttt{ON}, the unit tests of the library will be compiled. You will then
2013-09-24 22:02:09 +00:00
be able to launch the tests to make sure that they are running fine on your system.
2018-03-26 20:10:39 +00:00
\item[PROFILING\_ENABLED] If this variable is \texttt{ON}, the integrated profiler will collect data during the execution of the application.
This might be useful to see which part of the ReactPhysics3D
2013-10-06 16:46:32 +00:00
library takes time during its execution. This variable must be set to \texttt{OFF} when you compile
2018-03-26 20:10:39 +00:00
the final release of your application. You can find more information about the profiler in section \ref{sec:profiler}.
2013-09-24 22:02:09 +00:00
2018-03-26 20:10:39 +00:00
\item[LOGS\_ENABLED] Set this variable to \texttt{ON} if you want to enable the internal logger of ReactPhysics3D. Logs can be useful for debugging the application.
You can find more information about the logger in section \ref{sec:logger}.
2018-03-19 22:07:09 +00:00
2015-03-01 21:37:22 +00:00
\item[DOUBLE\_PRECISION\_ENABLED] If this variable is \texttt{ON}, the library will be compiled with double floating point precision.
Otherwise, the library will be compiled with single precision.
2013-09-24 22:02:09 +00:00
\end{description}
2016-04-09 20:52:25 +00:00
2013-03-18 21:14:43 +00:00
\section{Using ReactPhysics3D in your application}
2018-03-19 22:07:09 +00:00
TODO : Update this section with "make install" way
In order to use ReactPhysics3D in your own application, first build
the static library as described above to get the
2013-09-24 22:02:09 +00:00
static library file in the \texttt{lib/} folder. Then, in your code, you have to include
2015-03-01 21:37:22 +00:00
the ReactPhysics3D header file with the line: \\
2013-03-18 21:14:43 +00:00
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2016-04-14 20:11:19 +00:00
// Include the ReactPhysics3D header file
#include "reactphysics3d.h"
2013-09-24 22:02:09 +00:00
\end{lstlisting}
2016-04-09 20:52:25 +00:00
2013-09-24 22:02:09 +00:00
\vspace{0.6cm}
2013-03-18 21:14:43 +00:00
Note that the \texttt{reactphysics3d.h} header file can be found in the
2013-09-24 22:02:09 +00:00
\texttt{src/} folder of the library. Do not forget to add the
\texttt{src/} folder in your include directories in order that the
2013-03-18 21:14:43 +00:00
\texttt{reactphysics3d.h} file is accessible in your code. \\
2013-09-24 22:02:09 +00:00
Do not forget to also link your application with the ReactPhysics3D
2013-03-18 21:14:43 +00:00
static library. \\
2016-04-09 20:52:25 +00:00
2013-03-18 21:14:43 +00:00
Then, you should be able to compile your application using the
2013-09-24 22:02:09 +00:00
ReactPhysics3D library. \\
All the classes of the library are available in the \texttt{reactphysics3d} namespace or its shorter alias
2015-03-01 21:37:22 +00:00
\texttt{rp3d}. Therefore, you need to include this namespace into your code with the following declaration: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2016-04-14 20:11:19 +00:00
// Use the ReactPhysics3D namespace
using namespace reactphysics3d;
\end{lstlisting}
2013-09-24 22:02:09 +00:00
\vspace{0.6cm}
2015-03-01 21:37:22 +00:00
You can also take a look at the examples and the API documentation to get a better idea of how to use the
2013-03-18 21:14:43 +00:00
ReactPhysics3D library.
2018-03-06 18:29:16 +00:00
\subsection{Memory allocation}
When using the ReactPhysics3D library, you will probably have to allocate new objects but the library will also allocate some objects for you. ReactPhysics3D uses a simple rule:
all the memory that is allocated by yourself (using the C++ \emph{new} operator for instance) will have to be destroyed by you (with the corresponding \emph{delete} operator). However, if you
receive a pointer to an object from a call to a ReactPhysics3D method, you have not allocated memory for this object by yourself and therefore, you are not responsible to destroy it. ReactPhysics3D
will have to destroy it. \\
For instance, if you create a sphere collision shape with the following code: \\
\begin{lstlisting}
// Here memory is allocated by yourself
rp3d::SphereShape* sphereShape = new SphereShape(radius);
\end{lstlisting}
\vspace{0.6cm}
In this example, you have allocated memory for the collision shape and therefore, you have to destroy this object at the end as follows: \\
\begin{lstlisting}
delete sphereShape;
\end{lstlisting}
\vspace{0.6cm}
However, when you create a RigidBody with the following code: \\
\begin{lstlisting}
// Here memory is allocated by the ReactPhysics3D library
RigidBody* body = dynamicsWorld.createRigidBody(transform);
\end{lstlisting}
\vspace{0.6cm}
Here, the ReactPhysics3D library has allocated the memory for the rigid body and has returned a pointer so that you can use it. Because you have not allocated memory by yourself here, you must not
destroy the rigid body. The library is responsible to do it. \\
2013-03-18 21:14:43 +00:00
2015-03-01 21:37:22 +00:00
\section{The Collision World}
2018-03-19 22:07:09 +00:00
\label{sec:collisionworld}
2015-03-01 21:37:22 +00:00
There are two main ways to use ReactPhysics3D. The first one is to create bodies that you have to manually move so that you can test collision between them. To do this,
2018-03-19 22:07:09 +00:00
you need to create a collision world with several collision bodies in it. The second way is to create bodies and let ReactPhysics3D simulate their motions automatically using the
physics. This is done by creating rigid bodies in a dynamics world instead. In summary, a collision world is used to simply test collision between bodies that you have to manually move
and a dynamics world is used to create bodies that will be automatically moved using collisions, joints and forces. \\
2015-03-01 21:37:22 +00:00
2018-03-19 22:07:09 +00:00
The \texttt{CollisionWorld} class represents a collision world in the ReactPhysics3D library.
2015-03-01 21:37:22 +00:00
\subsection{Creating the Collision World}
If you only have to test collision between bodies, the first thing to do is to create an instance of the \texttt{CollisionWorld} class. \\
2018-03-19 22:07:09 +00:00
Here is how to create a collision world: \\
2015-03-01 21:37:22 +00:00
\begin{lstlisting}
// Create the collision world
rp3d::CollisionWorld world;
\end{lstlisting}
2018-03-19 22:07:09 +00:00
\subsubsection{World settings}
When you create a world as in the previous example, it will have default settings. If you want to customize some settings, you need to create
a \texttt{WorldSettings} object and give it in paramater when you create your world as in the following example: \\
\begin{lstlisting}
2018-03-26 20:10:39 +00:00
// Create the world settings
rp3d::WorldSettings settings;
settings.nbVelocitySolverIterations = 20;
settings.isSleepingEnabled = false;
// Create the world with your settings
rp3d::CollisionWorld world(settings);
2018-03-19 22:07:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
2018-03-26 20:10:39 +00:00
The settings are copied into the world at its creation. Therefore, changing the values of your \texttt{WorldSettings} instance after the world constructor call
will not have any effects. However, some methods are available to change settings after the world creation. You can take a look at the API documentation to see what
world settings can be changed.
2018-03-19 22:07:09 +00:00
2015-03-01 21:37:22 +00:00
\subsection{Destroying the Collision World}
Do not forget to destroy the \texttt{CollisionWorld} instance at the end of your program in order to release the allocated memory. If the object has been created
statically, it will be destroyed automatically at the end of the scope in which it has been created. If the object has been created dynamically (using the \texttt{new}
2015-08-12 16:14:40 +00:00
operator), you need to destroy it with the \texttt{delete} operator. \\
When the \texttt{CollisionWorld} is destroyed, all the bodies that have been added into it and that have not been destroyed already will be destroyed.
Therefore, the pointers to the bodies of the world will become invalid after the existence of their \texttt{CollisionWorld}.
2015-03-01 21:37:22 +00:00
2018-03-06 18:29:16 +00:00
\subsection{Queries on the Collision World}
2018-03-19 22:07:09 +00:00
Once your collision world has been created, you can add collision bodies and move them around manually. Now there are some queries that you can perform on the
collision world. Here are the main methods that you can use:
2018-03-06 18:29:16 +00:00
\begin{description}
2018-03-19 22:07:09 +00:00
\item[testOverlap()] Those methods can be used to test whether the collision shapes of two bodies overlap or not. You can use this if you just want to
know if bodies are colliding but your are not interested in the contact information. This method can be called on a \texttt{CollisionWorld}.
\item[testCollision()] Those methods will give you the collision information (contact points, normals, ...) for colliding bodies.
This method can be called on a \texttt{CollisionWorld}.
\item[testAABBOverlap()] Those methods will test whether AABBs of bodies overlap. This is faster than \texttt{testOverlap()} but less precise because it only
use the AABBs and not the actual collision shapes of the bodies. This method can be called on a \texttt{CollisionWorld}.
\item[testPointInside()] This method will tell if you if a 3D point is inside a given \texttt{CollisionBody} or \texttt{ProxyShape}.
2018-03-06 18:29:16 +00:00
\end{description}
2018-03-19 22:07:09 +00:00
Take a look at the API documentation for more information about those methods.
2015-03-01 21:37:22 +00:00
\section{Collision Bodies}
2018-03-19 22:07:09 +00:00
Once the collision world has been created, you can create collision bodies into the world. A collision body represents an object in the collision world.
It has a position, an orientation and one or more collision shapes. It has to be moved manually in the collision world. You can then
test collisions between the collision bodies of the world. In ReactPhysics3D, the \texttt{CollisionBody} class is used to describe a collision body. \\
2015-03-01 21:37:22 +00:00
2018-03-19 22:07:09 +00:00
If you do not want to simply test collision between your bodies but want them to move automatically according to the physics, you should use rigid bodies in a
dynamics world instead. See section \ref{sec:dynamicsworld} for more information about the dynamics world and section \ref{sec:rigidbody} if you would like to know more
about the rigid bodies.
2015-03-01 21:37:22 +00:00
\subsection{Creating a Collision Body}
2018-03-19 22:07:09 +00:00
In order to create a collision body, you need to specify its transform. The transform describes the initial
2015-03-01 21:37:22 +00:00
position and orientation of the body in the world. You need to create an instance of the \texttt{Transform} class with a vector describing the
initial position and a quaternion for the initial orientation of the body. \\
In order to test collision between your body and other bodies in the world, you need to add one or several collision shapes to your body.
Take a look at section \ref{sec:collisionshapes} to learn about the different collision shapes and how to create them. \\
2018-03-19 22:07:09 +00:00
You need to call the \texttt{CollisionWorld::createCollisionBody()} method to create a collision body in the world previously created. This method will return a pointer to the instance
2015-03-01 21:37:22 +00:00
of the \texttt{CollisionBody} class that has been created internally. You will then be able to use that pointer to get or set values of the body. \\
2018-03-19 22:07:09 +00:00
You can see in the following code how to create a collision body in the world. \\
2015-03-01 21:37:22 +00:00
\begin{lstlisting}
// Initial position and orientation of the collision body
rp3d::Vector3 initPosition(0.0, 3.0, 0.0);
rp3d::Quaternion initOrientation = rp3d::Quaternion::identity();
rp3d::Transform transform(initPosition, initOrientation);
// Create a collision body in the world
rp3d::CollisionBody* body;
body = world.createCollisionBody(transform);
\end{lstlisting}
\subsection{Moving a Collision Body}
2018-03-19 22:07:09 +00:00
A collision body has to be moved manually in the world. To do that, you need to use the \texttt{CollisionBody::setTransform()} method to set a new position and new
orientation to the body. \\
2015-03-01 21:37:22 +00:00
\begin{lstlisting}
// New position and orientation of the collision body
rp3d::Vector3 position(10.0, 3.0, 0.0);
rp3d::Quaternion orientation = rp3d::Quaternion::identity();
rp3d::Transform newTransform(position, orientation);
// Move the collision body
body->setTransform(newTransform);
\end{lstlisting}
\subsection{Destroying a Collision Body}
\begin{sloppypar}
2018-03-19 22:07:09 +00:00
In order to destroy a collision body from the world, you need to use the \texttt{CollisionWorld::destroyCollisionBody()} method. You need to use the pointer to the body you
2016-04-09 20:52:25 +00:00
want to destroy in argument. Note that after calling that method, the pointer will not be valid anymore and therefore, you should not use it. \\
2015-03-01 21:37:22 +00:00
\end{sloppypar}
2018-03-19 22:07:09 +00:00
Here is how to destroy a collision body: \\
2015-03-01 21:37:22 +00:00
\begin{lstlisting}
// Here, world is an instance of the CollisionWorld class
// and body is a CollisionBody* pointer
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Destroy the collision body and remove it from the world
world.destroyCollisionBody(body);
\end{lstlisting}
\section{The Dynamics World}
\label{sec:dynamicsworld}
2018-03-19 22:07:09 +00:00
The collision world of the previous section is used to manually move the bodies and check for collision between them. On the other side, a dynamics world
2015-03-01 21:37:22 +00:00
is used to automatically simulate the motion of your bodies using the physics. You do not have to move the bodies manually (but you still can if needed).
2018-03-19 22:07:09 +00:00
The dynamics world will contain the bodies and joints that you create. You will then be able to run your simulation across time by updating the world at each frame.
The \texttt{DynamicsWorld} class (which inherits from the \texttt{CollisionWorld} class) represents a dynamics world in the ReactPhysics3D library.
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
\subsection{Creating the Dynamics World}
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
The first thing you have to do when you want to simulate the dynamics of rigid bodies in time is to create an instance
2015-08-13 17:12:51 +00:00
of the \texttt{DynamicsWorld}. You need to specify the gravity acceleration vector (in $m / s^2$) in the world as parameter. Note that gravity is
activated by default when you create the world. \\
2013-09-30 21:55:11 +00:00
2018-03-19 22:07:09 +00:00
Here is how to create the dynamics world: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Gravity vector
rp3d::Vector3 gravity(0.0, -9.81, 0.0);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Create the dynamics world
2015-08-13 17:12:51 +00:00
rp3d::DynamicsWorld world(gravity);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
2015-03-01 21:37:22 +00:00
\subsection{Customizing the Dynamics World}
2013-09-24 22:02:09 +00:00
2013-09-30 21:55:11 +00:00
\subsubsection{Solver parameters}
2018-03-19 22:07:09 +00:00
ReactPhysics3D uses an iterative solver to simulate the contacts and joints. For contacts, there is a unique velocity solver and for joints there is a velocity and a
2013-09-30 21:55:11 +00:00
position solver. By default, the number of iterations of the velocity solver is 10 and the number of iterations for the position solver is 5. It is possible to
2013-10-06 16:46:32 +00:00
change the number of iterations for both solvers. \\
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
To do this, you need to use the following two methods: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Change the number of iterations of the velocity solver
world.setNbIterationsVelocitySolver(15);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Change the number of iterations of the position solver
world.setNbIterationsPositionSolver(8);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\vspace{0.6cm}
Increasing the number of iterations of the solvers will make the simulation more precise but also more expensive to compute. Therefore, you need to change
those values only if needed.
\subsubsection{Sleeping}
2013-10-06 16:46:32 +00:00
\label{sec:sleeping}
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
The purpose of the sleeping technique is to deactivate resting bodies so that they are not simulated anymore. This is used to save computation time because simulating
many bodies is costly.
A sleeping body (or group of sleeping bodies) is awaken as soon as another body collides with it or a joint in which it is involed is enabled. The sleeping technique
is enabled by default. You can disable it using the following method: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Disable the sleeping technique
world.enableSleeping(false);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\vspace{0.6cm}
2015-03-01 21:37:22 +00:00
Note that it is not recommended to disable the sleeping technique because the simulation might become slower. It is also possible to deactivate the sleeping technique on a
2013-10-06 16:46:32 +00:00
per body basis. See section \ref{sec:rigidbodysleeping} for more information. \\
2013-09-30 21:55:11 +00:00
2013-10-06 16:46:32 +00:00
\begin{sloppypar}
2015-03-01 21:37:22 +00:00
A body is put to sleep when its linear and angular velocity stay under a given velocity threshold for a certain amount of time (one second by default). It is possible to
2018-03-19 22:07:09 +00:00
change the linear and angular velocity thresholds using the two methods \texttt{DynamicsWorld::setSleepLinearVelocity()} and \texttt{DynamicsWorld::setSleepAngularVelocity()}.
2015-03-01 21:37:22 +00:00
Note that the velocities must
be specified in meters per second. You can also change the amount of time (in seconds) the velocity of a body needs to stay under the threshold to be considered
sleeping. To do this, use the
2013-10-06 16:46:32 +00:00
\texttt{DynamicsWorld::setTimeBeforeSleep()} method.
\end{sloppypar}
2016-04-09 20:52:25 +00:00
2015-03-01 21:37:22 +00:00
\subsection{Updating the Dynamics World}
2018-03-06 18:29:16 +00:00
\label{sec:updatingdynamicsworld}
2013-09-24 22:02:09 +00:00
2015-08-13 17:12:51 +00:00
The \texttt{DynamicsWorld} is used to simulate physics through time. It has to be updated each time you want to simulate a step forward in time. Most of the time,
you want to update the world right before rendering a new frame in a real-time application. \\
2013-09-30 21:55:11 +00:00
2015-08-13 17:12:51 +00:00
To update the physics world, you need to use the \texttt{DynamicsWorld::update()} method. This method will perform collision detection and update the
position and orientation of the bodies and joints. After updating the world, you will be able to get the new position and orientation of your bodies for the next
frame to render. This method requires a \emph{timeStep} parameter. This is the amount of time you want to advance the physics simulation (in seconds). \\
2013-09-30 21:55:11 +00:00
2015-08-13 17:12:51 +00:00
The smaller the time step you pick, the more precise the simulation will be but it can also be more expensive to compute. For a real-time application, you
probably want a time step of at most $\frac{1}{60}$ seconds to
have at least a 60 Hz framerate. Most of the time, physics engines prefer to work with a constant time step. It means that you should always call
the \texttt{DynamicsWorld::update()} method with the same time step parameter. You do not want to use the time between two frames as your time step because it will
not be constant. \\
2013-09-30 21:55:11 +00:00
2015-08-13 17:12:51 +00:00
You can use the following technique. First, you choose a constant time step for the physics. Let say the time step is $\frac{1}{60}$ seconds. Then, at each frame,
you compute the time difference between the current frame and the previous one and you accumulate this difference in a variable called \emph{accumulator}. The accumulator
is initialized to zero at the beginning of your application and is updated at each frame. The idea is to divide the time in the accumulator in several constant time steps.
For instance, if your accumulator contains $0.145$ seconds, it means that we can take $8$ physics steps of $\frac{1}{60}$ seconds during the current frame. Note that
$0.012$ seconds will remain in the accumulator and will probably be used in the next frame. As you can see, multiple physics steps can be taken at each frame. It is
important to understand that each call to the \texttt{DynamicsWorld::update()} method is done using a constant time step that is not varying with the framerate. \\
Here is what the code looks like at each frame: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-08-13 17:12:51 +00:00
// Constant physics time step
const float timeStep = 1.0 / 60.0;
2013-09-30 21:55:11 +00:00
2015-08-13 17:12:51 +00:00
// Get the current system time
long double currentFrameTime = getCurrentSystemTime();
2016-04-09 20:52:25 +00:00
2015-08-13 17:12:51 +00:00
// Compute the time difference between the two frames
long double deltaTime = currentFrameTime - previousFrameTime;
// Update the previous time
previousFrameTime = currentFrameTime;
2016-04-09 20:52:25 +00:00
// Add the time difference in the accumulator
2015-08-13 17:12:51 +00:00
accumulator += mDeltaTime;
// While there is enough accumulated time to take
// one or several physics steps
while (accumulator >= timeStep) {
2013-09-30 21:55:11 +00:00
2015-08-13 17:12:51 +00:00
// Update the Dynamics world with a constant time step
dynamicsWorld->update(timeStep);
2013-09-30 21:55:11 +00:00
2015-08-13 17:12:51 +00:00
// Decrease the accumulated time
accumulator -= timeStep;
}
2016-04-09 20:52:25 +00:00
2015-08-13 17:12:51 +00:00
\end{lstlisting}
2016-04-09 20:52:25 +00:00
2018-03-06 18:29:16 +00:00
\vspace{0.6cm}
2018-03-19 22:07:09 +00:00
A nice article to read about this is the one from Glenn Fiedler at \url{https://gafferongames.com/post/fix_your_timestep/}.
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
\subsection{Destroying the Dynamics World}
2013-09-24 22:02:09 +00:00
2015-08-12 16:14:40 +00:00
Do not forget to destroy the \texttt{DynamicsWorld} instance at the end of your program in order to release the allocated memory. If the object has been created
statically, it will automatically be destroyed at the end of the scope in which it has been created. If the object has been created dynamically (using the
\texttt{new} operator), you need to destroy it with the \texttt{delete} operator. \\
When the \texttt{DynamicsWorld} is destroyed, all the bodies and joints that have been added into it and that have not been destroyed already will be destroyed.
Therefore, the pointers to the bodies and joints of the world will become invalid after the existence of their \texttt{DynamicsWorld}.
2013-09-30 21:55:11 +00:00
2013-09-24 22:02:09 +00:00
\section{Rigid Bodies}
2015-03-01 21:37:22 +00:00
\label{sec:rigidbody}
2013-09-24 22:02:09 +00:00
2018-03-19 22:07:09 +00:00
Once the dynamics world has been created, you can create rigid bodies into the world. A rigid body represents an object that you want to simulate in the world.
It has a mass, a position, an orientation and one or several collision shapes. The dynamics world will compute collisions between the bodies and will update its position
2015-03-01 21:37:22 +00:00
and orientation accordingly at each time step. You can also create joints between the bodies in the world. In ReactPhysics3D, the \texttt{RigidBody} class
2018-03-19 22:07:09 +00:00
(which inherits from the \texttt{CollisionBody} class) is used to describe a rigid body.
2013-09-30 21:55:11 +00:00
2013-09-24 22:02:09 +00:00
\subsection{Creating a Rigid Body}
2018-03-19 22:07:09 +00:00
In order to create a rigid body, you need to specify its transform. The transform describes the initial
2015-03-01 21:37:22 +00:00
position and orientation of the body in the world. You need to create an instance of the \texttt{Transform} class with a vector describing the
2013-09-30 21:55:11 +00:00
initial position and a quaternion for the initial orientation of the body. \\
2018-03-19 22:07:09 +00:00
You have to call the \texttt{DynamicsWorld::createRigidBody()} method to create a rigid body in the world previously created. This method will return a pointer to the
2015-03-01 21:37:22 +00:00
instance of the \texttt{RigidBody} object that has been created internally. You will then be able to use that pointer to get or set values of the body. \\
2013-09-30 21:55:11 +00:00
2018-03-19 22:07:09 +00:00
You can see in the following code how to create a rigid body in your world: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Initial position and orientation of the rigid body
rp3d::Vector3 initPosition(0.0, 3.0, 0.0);
rp3d::Quaternion initOrientation = rp3d::Quaternion::identity();
rp3d::Transform transform(initPosition, initOrientation);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Create a rigid body in the world
rp3d::RigidBody* body;
body = dynamicsWorld.createRigidBody(transform);
\end{lstlisting}
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
\vspace{0.6cm}
2013-09-30 21:55:11 +00:00
2018-03-19 22:07:09 +00:00
Once your rigid body has been created in the world, you need to add one or several collision shapes to it. Take a look at section \ref{sec:collisionshapes} to learn
2015-03-01 21:37:22 +00:00
about the different collision shapes and how to create them. \\
2013-09-30 21:55:11 +00:00
2013-09-24 22:02:09 +00:00
\subsection{Customizing a Rigid Body}
2018-03-19 22:07:09 +00:00
Once a rigid body has been created, you can change some of its properties.
2013-10-06 16:46:32 +00:00
2015-03-01 21:37:22 +00:00
\subsubsection{Type of a Rigid Body (static, kinematic or dynamic)}
2013-10-06 16:46:32 +00:00
\begin{sloppypar}
2015-03-01 21:37:22 +00:00
There are three types of bodies: \emph{static}, \emph{kinematic} and \emph{dynamic}. A \emph{static} body has infinite mass, zero velocity but its position can be
changed manually. Moreover, a static body does not collide with other static or kinematic bodies. On the other side, a \emph{kinematic} body has infinite mass, its velocity can be
changed manually and its position is computed by the physics engine. A kinematic body does not collide with other static or kinematic bodies. Finally, A \emph{dynamic} body
has non-zero mass, non-zero velocity determined by forces and its position is determined by the physics engine. Moreover, a dynamic body can collide with other dynamic, static or
kinematic bodies. \\
2013-10-06 16:46:32 +00:00
\end{sloppypar}
2015-03-01 21:37:22 +00:00
When you create a new body in the world, it is of dynamic type by default. You can change the type of the body using the \texttt{CollisionBody::setType()}
method as follows:\\
2013-10-06 16:46:32 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Change the type of the body to Kinematic
body->setType(KINEMATIC);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\subsubsection{Gravity}
By default, all the rigid bodies with react to the gravity force of the world. If you do not want the gravity to be applied to a given body, you can disable
2018-03-19 22:07:09 +00:00
it using the \texttt{RigidBody::enableGravity()} method as in the following example: \\
2013-10-06 16:46:32 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Disable gravity for this body
rigidBody->enableGravity(false);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\subsubsection{Material of a Rigid Body}
2018-03-19 22:07:09 +00:00
The material of a rigid body is used to describe the physical properties it is made of. This is represented by the \texttt{Material} class. Each body that
2013-10-06 16:46:32 +00:00
you create will have a default material. You can get the material of the rigid body using the \texttt{RigidBody::getMaterial()} method. Then, you will be able to change some
properties. \\
For instance, you can change the bounciness of the rigid body. The bounciness is a value between 0 and 1. The value 1 is used for a very bouncy object and the value 0 means that
the body will not be bouncy at all. To change the bounciness of the material, you can use the \texttt{Material::setBounciness()} method. \\
You are also able to change the friction coefficient of the body. This value needs to be between 0 and 1. If the value is 0, no friction will be applied when the body is in contact with
another body. However, if the value is 1, the friction force will be high. You can change the friction coefficient of the material with the
\texttt{Material::setFrictionCoefficient()} method. \\
2016-04-09 20:52:25 +00:00
You can use the material to add rolling resistance to a rigid body. Rolling resistance can be used to stop
2018-03-19 22:07:09 +00:00
a rolling object on a flat surface for instance. You should use this only with SphereShape or
CapsuleShape collision shapes. By default, rolling resistance is zero but you can
2016-04-09 20:52:25 +00:00
set a positive value using the \texttt{Material::setRollingResistance()} method to increase resistance. \\
2018-03-19 22:07:09 +00:00
Here is how to get the material of a rigid body and how to modify some of its properties: \\
2013-10-06 16:46:32 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Get the current material of the body
rp3d::Material& material = rigidBody->getMaterial();
2013-10-06 16:46:32 +00:00
2015-03-01 21:37:22 +00:00
// Change the bounciness of the body
material.setBounciness(rp3d::decimal(0.4));
2013-10-06 16:46:32 +00:00
2015-03-01 21:37:22 +00:00
// Change the friction coefficient of the body
material.setFrictionCoefficient(rp3d::decimal(0.2));
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\subsubsection{Velocity Damping}
\begin{sloppypar}
2015-03-01 21:37:22 +00:00
Damping is the effect of reducing the velocity of the rigid body during the simulation to simulate effects like air friction for instance. By default, no damping
is applied. However, you can choose to damp the linear or/and the angular velocity of a rigid body. For instance, without angular damping a pendulum will never come
to rest. You need to use the \texttt{RigidBody::setLinearDamping()} and \texttt{RigidBody::setAngularDamping()} methods to change the damping values. The damping
value has to be positive and a value of zero means no damping at all.
2013-10-06 16:46:32 +00:00
\end{sloppypar}
\subsubsection{Sleeping}
\label{sec:rigidbodysleeping}
2015-03-01 21:37:22 +00:00
As described in section \ref{sec:sleeping}, the sleeping technique is used to disable the simulation of resting bodies. By default, the bodies are allowed to sleep
2018-03-19 22:07:09 +00:00
when they come to rest. However, if you do not want a given body to be put to sleep, you can use the \texttt{Body::setIsAllowedToSleep()} method as in the next example: \\
2013-10-06 16:46:32 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// This rigid body cannot sleep
2016-04-09 20:52:25 +00:00
rigidBody->setIsAllowedToSleep(false);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\subsubsection{Applying Force or Torque to a Rigid Body}
2018-03-19 22:07:09 +00:00
During the simulation, you can apply a force or a torque to a given rigid body. This force can be applied to the center of mass of the rigid body by using the
2015-03-01 21:37:22 +00:00
\texttt{RigidBody::applyForceToCenter()} method. You need to specify the force vector (in Newton) as a parameter. If the force is applied to the center of mass, no
2013-10-06 16:46:32 +00:00
torque will be created and only the linear motion of the body will be affected. \\
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Force vector (in Newton)
rp3d::Vector3 force(2.0, 0.0, 0.0);
2016-04-09 20:52:25 +00:00
2015-03-01 21:37:22 +00:00
// Apply a force to the center of the body
rigidBody->applyForceToCenter(force);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\vspace{0.6cm}
\begin{sloppypar}
You can also apply a force to any given point (in world-space) using the \texttt{RigidBody::applyForce()} method. You need to specify the force vector (in Newton) and the point
(in world-space) where to apply the given force. Note that if the point is not the center of mass of the body, applying a force will generate some torque and therefore, the
angular motion of the body will be affected as well. \\
\end{sloppypar}
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Force vector (in Newton)
rp3d::Vector3 force(2.0, 0.0, 0.0);
2013-10-06 16:46:32 +00:00
2015-03-01 21:37:22 +00:00
// Point where the force is applied
rp3d::Vector3 point(4.0, 5.0, 6.0);
2016-04-09 20:52:25 +00:00
2015-03-01 21:37:22 +00:00
// Apply a force to the body
rigidBody->applyForce(force, point);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\vspace{0.6cm}
\begin{sloppypar}
It is also possible to apply a torque to a given body using the \texttt{RigidBody::applyTorque()} method. You simply need to specify the torque vector (in Newton $\cdot$ meter) as
2015-03-01 21:37:22 +00:00
in the following example: \\
2013-10-06 16:46:32 +00:00
\end{sloppypar}
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Torque vector
rp3d::Vector3 torque(0.0, 3.0, 0.0);
2013-10-06 16:46:32 +00:00
2015-03-01 21:37:22 +00:00
// Apply a torque to the body
rigidBody->applyTorque(torque);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
\vspace{0.6cm}
Note that when you call the previous methods, the specified force/torque will be added to the total force/torque applied to the rigid body and that at the end of each call to the
2015-03-01 21:37:22 +00:00
\texttt{DynamicsWorld::update()}, the total force/torque of all the rigid bodies will be reset to zero. Therefore, you need to call the previous methods during several frames
2013-10-06 16:46:32 +00:00
if you want the force/torque to be applied during a certain amount of time.
2013-09-30 21:55:11 +00:00
2013-09-24 22:02:09 +00:00
\subsection{Updating a Rigid Body}
2018-03-06 18:29:16 +00:00
When you call the \texttt{DynamicsWorld::update()} method, the collisions between the bodies are computed and the joints are evaluated. Then, the bodies positions
and orientations are updated accordingly. After calling this method, you can retrieve the updated position and orientation of each body to render it. To do that, you simply need to use the
\texttt{RigidBody::getTransform()} method to get the updated transform. This transform represents the current local-to-world-space transformation
of the body. \\
2013-09-30 21:55:11 +00:00
2018-03-06 18:29:16 +00:00
As described in section \ref{sec:updatingdynamicsworld}, at the end of a frame, there might still be some remaining time in the time accumulator. Therefore, you should not use the updated
transform directly for rendering but you need to perform some interpolation between the updated transform and the one from the previous frame to get a smooth real-time simulation.
2018-03-19 22:07:09 +00:00
First, you need to compute the interpolation factor as folows: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2018-03-06 18:29:16 +00:00
// Compute the time interpolation factor
decimal factor = accumulator / timeStep;
\end{lstlisting}
2013-09-30 21:55:11 +00:00
2018-03-06 18:29:16 +00:00
\vspace{0.6cm}
2017-08-01 10:39:20 +00:00
2018-03-06 18:29:16 +00:00
Then, you can use the \texttt{Transform::interpolateTransforms()} method to compute the linearly interpolated transform: \\
2017-08-01 10:39:20 +00:00
2018-03-06 18:29:16 +00:00
\begin{lstlisting}
// Compute the interpolated transform of the rigid body
rp3d::Transform interpolatedTransform = Transform::interpolateTransforms(prevTransform, currTransform, factor);
\end{lstlisting}
2017-08-01 10:39:20 +00:00
2018-03-06 18:29:16 +00:00
\vspace{0.6cm}
The following code is the one from section \ref{sec:updatingdynamicsworld} for the physics simulation loop but with the update of a given rigid body. \\
2017-08-01 10:39:20 +00:00
\begin{lstlisting}
2018-03-06 18:29:16 +00:00
// Constant physics time step
const float timeStep = 1.0 / 60.0;
// Get the current system time
long double currentFrameTime = getCurrentSystemTime();
// Compute the time difference between the two frames
long double deltaTime = currentFrameTime - previousFrameTime;
// Update the previous time
previousFrameTime = currentFrameTime;
// Add the time difference in the accumulator
accumulator += mDeltaTime;
// While there is enough accumulated time to take
// one or several physics steps
while (accumulator >= timeStep) {
// Update the Dynamics world with a constant time step
dynamicsWorld->update(timeStep);
// Decrease the accumulated time
accumulator -= timeStep;
}
// Compute the time interpolation factor
decimal factor = accumulator / timeStep;
// Get the updated transform of the body
rp3d::Transform currTransform = body->getTransform();
// Compute the interpolated transform of the rigid body
rp3d::Transform interpolatedTransform = Transform::interpolateTransforms(prevTransform, currTransform, factor);
// Now you can render your body using the interpolated transform here
// Update the previous transform
prevTransform = currTranform;
2017-08-01 10:39:20 +00:00
\end{lstlisting}
2013-09-30 21:55:11 +00:00
\vspace{0.6cm}
2018-03-06 18:29:16 +00:00
If you need the array with the corresponding $4 \times 4$ OpenGL transformation matrix for rendering, you can use the \texttt{Transform::getOpenGLMatrix()} method as in the
2015-03-01 21:37:22 +00:00
following code: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2018-03-06 18:29:16 +00:00
// Get the OpenGL matrix array of the transform
float matrix[16];
transform.getOpenGLMatrix(matrix);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
2018-03-06 18:29:16 +00:00
\vspace{0.6cm}
A nice article to read about this time interpolation is the one from Glenn Fiedler at \url{https://gafferongames.com/post/fix_your_timestep/}.
2013-09-24 22:02:09 +00:00
\subsection{Destroying a Rigid Body}
2013-09-30 21:55:11 +00:00
\begin{sloppypar}
It is really simple to destroy a rigid body. You simply need to use the \texttt{DynamicsWorld::destroyRigidBody()} method. You need to use the pointer to the body you
2015-03-01 21:37:22 +00:00
want to destroy as a parameter. Note that after calling that method, the pointer will not be valid anymore and therefore, you should not use it. Note that you must
destroy all the rigid bodies at the end of the simulation before you destroy the world. When you destroy a rigid body that was part of a joint, that joint will be
automatically destroyed as well. \\
2013-09-30 21:55:11 +00:00
\end{sloppypar}
2015-03-01 21:37:22 +00:00
Here is how to destroy a rigid body: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Here, world is an instance of the DynamicsWorld class
// and body is a RigidBody* pointer
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Destroy the rigid body
world.destroyRigidBody(body);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
2013-09-24 22:02:09 +00:00
\section{Collision Shapes}
2013-09-30 21:55:11 +00:00
\label{sec:collisionshapes}
2013-09-24 22:02:09 +00:00
2018-03-19 22:07:09 +00:00
Once you have created a collision body or a rigid body in the world, you need to add one or more collision shapes into it so that it is able to collide with other bodies.
2013-09-24 22:02:09 +00:00
This section describes all the collision shapes available in the ReactPhysics3D library and how to use them. \\
2018-03-19 22:07:09 +00:00
The collision shapes are also the way to represent the mass of a Rigid Body. Whenever you add a collision shape to a rigid body, you need to specify the mass of the shape.
Then the rigid body will recompute its total mass, its center of mass and its inertia tensor taking into account all its collision shapes. Therefore, you do not have to compute
2015-03-01 21:37:22 +00:00
those things by yourself. However, if needed, you can also specify your own center of mass or inertia tensor. Note that the inertia tensor is a $3 \times 3$ matrix describing
how the mass is distributed inside the rigid body which will be used to calculate its rotation. The inertia tensor depends on the mass and the shape of the body. \\
2013-09-24 22:02:09 +00:00
\subsection{Box Shape}
\begin{figure}[h]
\centering
\includegraphics{boxshape.png}
\label{fig:boxshape}
\end{figure}
2018-03-19 22:07:09 +00:00
The \texttt{BoxShape} class describes a box collision shape. The box is aligned with the shape local X, Y and Z axis.
2013-09-24 22:02:09 +00:00
In order to create a box shape, you only need to specify the three half extents dimensions of the box in the three X, Y and Z directions. \\
2015-03-01 21:37:22 +00:00
For instance, if you want to create a box shape with dimensions of 4 meters, 6 meters and 10 meters along the X, Y and Z axis respectively, you need to use the
following code: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Half extents of the box in the x, y and z directions
const rp3d::Vector3 halfExtents(2.0, 3.0, 5.0);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Create the box shape
const rp3d::BoxShape boxShape(halfExtents);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
\subsection{Sphere Shape}
\begin{figure}[h]
\centering
\includegraphics{sphereshape.png}
\label{fig:sphereshape}
\end{figure}
2015-03-01 21:37:22 +00:00
The \texttt{SphereShape} class describes a sphere collision shape centered at the origin of the shape local space. You only need to specify the radius of the sphere to create it. \\
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
For instance, if you want to create a sphere shape with a radius of 2 meters, you need to use the following code: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the sphere shape with a radius of 2m
const rp3d::SphereShape sphereShape(2.0);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
\subsection{Capsule Shape}
\begin{figure}[h]
\centering
\includegraphics{capsuleshape.png}
\label{fig:capsuleshape}
\end{figure}
2018-03-19 22:07:09 +00:00
The \texttt{CapsuleShape} class describes a capsule collision shape around the local Y axis and centered at the origin of the shape local-space. It is the convex hull of two
2013-09-24 22:02:09 +00:00
spheres. It can also be seen as an elongated sphere. In order to create it, you only need to specify the radius of the two spheres and the height of the
capsule (distance between the centers of the two spheres). \\
2015-03-01 21:37:22 +00:00
For instance, if you want to create a capsule shape with a radius of 1 meter and the height of 2 meters, you need to use the following code: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the capsule shape
const rp3d::CapsuleShape capsuleShape(1.0, 2.0);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
\subsection{Convex Mesh Shape}
\begin{figure}[h]
\centering
\includegraphics{convexshape.png}
\label{fig:convexshape}
\end{figure}
2018-03-06 18:29:16 +00:00
The \texttt{ConvexMeshShape} class can be used to describe the shape of a convex mesh. In order to create a convex mesh shape, you first need to create an array of \texttt{PolygonFace} to describe each face
of your mesh. You also need to have an array with the vertices coordinates and an array with the vertex indices of each face of you mesh. Then, you have to create a \texttt{PolygonVertexArray} with your
vertices coordinates and indices array. You also need to specify your array of \texttt{PolygonFace}. Then, you have to create a \texttt{PolyhedronMesh} with your \texttt{PolygonVertexArray}.
Once this is done, you can create the \texttt{ConvexMeshShape} by passing your \texttt{PolyhedronMesh} in paramater. \\
2015-03-01 21:37:22 +00:00
2018-03-06 18:29:16 +00:00
The following example shows how to create a convex mesh shape. In this example, we create a cube as a convex mesh shape. Of course, this is only for the example.
If you really need a cube collision shape, you should use the \texttt{BoxShape} instead. \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2018-03-06 18:29:16 +00:00
// Array with the vertices coordinates of the convex mesh
float vertices[24];
vertices[0] = -3; vertices[1] = -3; vertices[2] = 3;
vertices[3] = 3; vertices[4] = -3; vertices[5] = 3;
vertices[6] = 3; vertices[7] = -3; vertices[8] = -3;
vertices[9] = -3; vertices[10] = -3; vertices[11] = -3;
vertices[12] = -3; vertices[13] = 3; vertices[14] = 3;
vertices[15] = 3; vertices[16] = 3; vertices[17] = 3;
vertices[18] = 3; vertices[19] = 3; vertices[20] = -3;
vertices[21] = -3; vertices[22] = 3; vertices[23] = -3;
// Array with the vertices indices for each face of the mesh
int indices[24];
indices[0]=0; indices[1]=3; indices[2]=2; indices[3]=1;
indices[4]=4; indices[5]=5; indices[6]=6; indices[7]=7;
indices[8]=0; indices[9]=1; indices[10]=5; indices[11]=4;
indices[12]=1; indices[13]=2; indices[14]=6; indices[15]=5;
indices[16]=2; indices[17]=3; indices[18]=7; indices[19]=6;
indices[20]=0; indices[21]=4; indices[22]=7; indices[23]=3;
// Description of the six faces of the convex mesh
polygonFaces = new rp3d::PolygonVertexArray::PolygonFace[6];
rp3d::PolygonVertexArray::PolygonFace* face = polygonFaces;
for (int f = 0; f < 6; f++) {
// First vertex of the face in the indices array
face->indexBase = f * 4;
// Number of vertices in the face
face->nbVertices = 4;
face++;
}
2013-09-24 22:02:09 +00:00
2018-03-06 18:29:16 +00:00
// Create the polygon vertex array
polygonVertexArray = new rp3d::PolygonVertexArray(8, vertices, 3 x sizeof(float),
indices, sizeof(int), 6, polygonFaces,
rp3d::PolygonVertexArray::VertexDataType::VERTEX_FLOAT_TYPE,
rp3d::PolygonVertexArray::IndexDataType::INDEX_INTEGER_TYPE);
2013-09-24 22:02:09 +00:00
2018-03-06 18:29:16 +00:00
// Create the polyhedron mesh
polyhedronMesh = new rp3d::PolyhedronMesh(polygonVertexArray);
2013-09-24 22:02:09 +00:00
2018-03-06 18:29:16 +00:00
// Create the convex mesh collision shape
convexMeshShape = new rp3d::ConvexMeshShape(polyhedronMesh);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
2018-03-06 18:29:16 +00:00
Note that the vertex coordinates and indices array are not copied and therefore, you need to make sure that they exist until the collision shape exists. This is also true for
the all the \texttt{PolygonFace}, the \texttt{PolygonVertexArray} and the \texttt{PolyhedronMesh}. \\
2013-09-24 22:02:09 +00:00
2018-03-06 18:29:16 +00:00
You need to make sure that the mesh you provide is indeed convex. Secondly, you should provide the simplest possible convex mesh. It means that you need to avoid
2018-03-19 22:07:09 +00:00
coplanar faces in your convex mesh shape. Coplanar faces have to be merged together. Remember that convex meshes are not limited to triangular faces, you can
2018-03-06 18:29:16 +00:00
create faces with more than three vertices. \\
When you specify the vertices for each face of your convex mesh, be careful with their order. The vertices of a face must be specified in counter clockwise order
as seen from the outside of your convex mesh. \\
2018-04-04 16:39:30 +00:00
You can also specify a scaling factor in the constructor when you create a \texttt{ConvexMeshShape}. All the vertices of your mesh will be scaled from the origin by this factor
when used in the collision shape. \\
2018-03-06 18:29:16 +00:00
Note that collision detection with a \texttt{ConvexMeshShape} is more expensive than with a \texttt{SphereShape} or a \texttt{CapsuleShape}. \\
2015-03-01 21:37:22 +00:00
2016-04-09 20:52:25 +00:00
\subsection{Concave Mesh Shape}
\begin{figure}[h]
\centering
\includegraphics{concavemeshshape.png}
\label{fig:concaveshape}
\end{figure}
The \texttt{ConcaveMeshShape} class can be used for a static concave triangular mesh. It can be used to describe an environment for
instance. Note that it cannot be used with a dynamic body that is allowed to move. Moreover, make sure to use a \texttt{ConcaveMeshShape} only
when you are not able to use a convex shape and also try to limit the number of triangles of that mesh because collision detection
with \texttt{ConcaveMeshShape} is quite expensive compared to convex shapes. \\
2018-03-06 18:29:16 +00:00
In order to create a concave mesh shape, you need to supply a pointer to a \texttt{TriangleMesh}. A \texttt{TriangleMesh} class
describes a mesh made of triangles. It may contain several parts (submeshes). Each part is a set of
2018-03-19 22:07:09 +00:00
triangles represented by a \texttt{TriangleVertexArray} object. A \texttt{TriangleVertexArray} represents
2016-04-09 20:52:25 +00:00
a continuous array of vertices and indexes for a triangular mesh. When you create a \texttt{TriangleVertexArray}, no data is copied
2018-03-06 18:29:16 +00:00
into the array. It only stores a pointer to the data. The idea is to allow the user to share vertices data between the physics engine and the rendering
2016-04-09 20:52:25 +00:00
part. Therefore, make sure that the data pointed by a \texttt{TriangleVertexArray} remains valid during the whole \texttt{TriangleVertexArray} life.
\\
2018-03-06 18:29:16 +00:00
The following code shows how to create a \texttt{TriangleVertexArray}: \\
2016-04-09 20:52:25 +00:00
\begin{lstlisting}
2016-04-14 20:11:19 +00:00
const int nbVertices = 8;
const int nbTriangles = 12;
float vertices[3 * nbVertices] = ...;
int indices[3 * nbTriangles] = ...;
rp3d::TriangleVertexArray* triangleArray =
new rp3d::TriangleVertexArray(nbVertices, vertices, 3 * sizeof(float), nbTriangles,
2018-03-06 18:29:16 +00:00
indices, 3 * sizeof(int),
2016-04-14 20:11:19 +00:00
rp3d::TriangleVertexArray::VERTEX_FLOAT_TYPE,
rp3d::TriangleVertexArray::INDEX_INTEGER_TYPE);
2016-04-09 20:52:25 +00:00
\end{lstlisting}
\vspace{0.6cm}
Now that we have a \texttt{TriangleVertexArray}, we need to create a \texttt{TriangleMesh} and add the \texttt{TriangleVertexArray}
2018-03-06 18:29:16 +00:00
into it as a subpart. Once this is done, we can create the actual \texttt{ConcaveMeshShape}. \\
2016-04-09 20:52:25 +00:00
\begin{lstlisting}
2016-04-14 20:11:19 +00:00
rp3d::TriangleMesh triangleMesh;
2016-04-09 20:52:25 +00:00
2016-04-14 20:11:19 +00:00
// Add the triangle vertex array to the triangle mesh
triangleMesh.addSubpart(triangleArray);
2016-04-09 20:52:25 +00:00
2016-04-14 20:11:19 +00:00
// Create the concave mesh shape
2018-03-06 18:29:16 +00:00
ConcaveMeshShape* concaveMesh = new rp3d::ConcaveMeshShape(&triangleMesh);
2016-04-09 20:52:25 +00:00
\end{lstlisting}
\vspace{0.6cm}
Note that the \texttt{TriangleMesh} object also needs to exist during the whole life of the collision shape because its
2018-03-06 18:29:16 +00:00
data is not copied into the collision shape. \\
When you specify the vertices for each triangle face of your mesh, be careful with the order of the vertices. They must be specified in counter clockwise order
as seen from the outside of your mesh. \\
2018-04-04 16:39:30 +00:00
You can also specify a scaling factor in the constructor when you create a \texttt{ConcaveMeshShape}. All the vertices of your mesh will be scaled from the origin by this factor
when used in the collision shape. \\
2018-03-06 18:29:16 +00:00
In the previous example, the vertex normals that are needed for collision detection are automatically computed. However, if you want to specify your own
vertex normals, you can do it by using another constructor for the \texttt{TriangleVertexArray}. \\
2016-04-09 20:52:25 +00:00
\subsection{Heightfield Shape}
\begin{figure}[h]
\centering
\includegraphics{heightfieldshape.png}
\label{fig:heightfieldshape}
\end{figure}
The \texttt{HeightFieldShape} is a collision shape that can be used to represent a static terrain for instance. You can
define a heightfield with a two dimensional grid that has a given height value at each point. \\
In order to create a \texttt{HeightFieldShape}, you need to have an array with all the height values of your field.
You can have height values of type int, float or double. You need to give the number of rows and columns of your two
dimensional grid. Note that the height values in your array must be organized such that the value at row
2018-03-19 22:07:09 +00:00
\texttt{indexRow} and column \texttt{indexColumn} is located at the following position in the array: \\
2016-04-09 20:52:25 +00:00
\begin{lstlisting}
2016-04-14 20:11:19 +00:00
heighFieldValues[indexRow * nbColumns + indexColumn]
2016-04-09 20:52:25 +00:00
\end{lstlisting}
\vspace{0.6cm}
Morevover, you need to provide the minimum and maximum height values of your height field. \\
Here is an example that shows how to create a \texttt{HeightFieldShape}: \\
\begin{lstlisting}
2016-04-14 20:11:19 +00:00
const int nbRows = 40;
const int nbColumns = 50;
float minHeight = 100;
float maxHeight = 500;
2016-04-09 20:52:25 +00:00
2016-04-14 20:11:19 +00:00
// Height values
float heightValues[nbRows * nbColumns] = ...;
2016-04-09 20:52:25 +00:00
2016-04-14 20:11:19 +00:00
// Create the heightfield collision shape
rp3d::HeightFieldShape = new rp3d::HeightFieldShape(nbColumns, nbRows, minHeight,
maxHeight, heightValues, rp3d::HeightFieldShape::HEIGHT_FLOAT_TYPE);
2016-04-09 20:52:25 +00:00
\end{lstlisting}
\vspace{0.6cm}
2018-03-19 22:07:09 +00:00
Note that the array of height values are not copied into the \texttt{HeightFieldShape}. Therefore, you need to make sure
they exist during the lifetime of the \texttt{HeightFieldShape} and you must not forget to release their memory when you
2016-04-09 20:52:25 +00:00
destroy the collision shape or at the end of your application. \\
2018-04-04 16:39:30 +00:00
You can also specify a scaling factor in the constructor when you create a \texttt{HeightFieldShape}. All the vertices of your mesh will be scaled from the origin by this factor
when used in the collision shape. \\
2016-04-09 20:52:25 +00:00
When creating a \texttt{HeightFieldShape}, the origin of the shape will be at the center of its bounding volume.
Therefore, if you create a \texttt{HeightFieldShape} with a minimum height of 100 and a maximum height of 500, the
maximum coordinates of the shape on the Y axis will be 200 and the minimum coordinates will be -200.
2015-03-01 21:37:22 +00:00
\subsection{Adding a Collision Shape to a body - The Proxy Shape concept}
\begin{sloppypar}
Now that you know how to create a collision shape, we will see how to add it to a given body. \\
2016-04-09 20:52:25 +00:00
First note that when you add a collision shape to a body, the shape will not be copied internally. You only give a
pointer to the shape in parameter. The shape must exist during the whole lifetime of the body. This way, you can
create a collision shape and reuse it for multiple bodies. You are also responsible to destroy the shape at the
end when the bodies are not used anymore. \\
2015-03-01 21:37:22 +00:00
2018-03-19 22:07:09 +00:00
In order to add a collision shape to a body, you need to use the \texttt{CollisionBody::addCollisionShape()} method for a collision body and the
\texttt{RigidBody::addCollisionShape()} method for a rigid body. You will have to provide the collision shape transform in parameter. This is the
transformation mapping the local-space of the collision shape to the local-space of the body. For a rigid body, you will also have to provide the
2015-03-01 21:37:22 +00:00
mass of the shape you want to add. As explained before, this is used to automatically compute the center of mass, total mass and inertia tensor of the body. \\
2018-03-19 22:07:09 +00:00
The \texttt{addCollisionShape()} method returns a pointer to a proxy shape. A proxy shape is what links a given collision shape to the body it has been added.
You can use the returned proxy shape to get or set parameters of the given collision shape in that particular body. This concept is also called \emph{fixture} in some
other physics engines. In ReactPhysics3D, a proxy shape is represented by the \texttt{ProxyShape} class. \\
2015-03-01 21:37:22 +00:00
The following example shows how to add a sphere collision shape with a given mass to a rigid body and also how to remove it from the body using the Proxy Shape pointer. \\
\end{sloppypar}
\begin{lstlisting}
// Create the sphere collision shape
rp3d::decimal radius = rp3d::decimal(3.0)
const rp3d::BoxShape shape(radius);
// Transform of the collision shape
// Place the shape at the origin of the body local-space
rp3d::Transform transform = rp3::Transform::identity();
// Mass of the collision shape (in kilograms)
rp3d::decimal mass = rp3d::decimal(4.0);
// Add the collision shape to the rigid body
rp3d::ProxyShape* proxyShape;
2016-04-09 20:52:25 +00:00
proxyShape = body->addCollisionShape(&shape, transform, mass);
2015-03-01 21:37:22 +00:00
// If you want to remove the collision shape from the body
// at some point, you need to use the proxy shape
2016-04-09 20:52:25 +00:00
body->removeCollisionShape(proxyShape);
2015-03-01 21:37:22 +00:00
\end{lstlisting}
\vspace{0.6cm}
2018-03-19 22:07:09 +00:00
As you can see, you can use the \texttt{removeCollisionShape()} method to remove a collision shape from a body by using the proxy shape. Note that
after removing a collision shape, the corresponding proxy shape pointer will not be valid anymore. It is not necessary to manually remove all the collision shapes from
2015-03-01 21:37:22 +00:00
a body at the end of your application. They will automatically be removed when you destroy the body.
\subsection{Collision filtering}
\label{sec:collisionfiltering}
2016-04-09 20:52:25 +00:00
2015-03-01 21:37:22 +00:00
By default all the collision shapes of all your bodies are able to collide with each other in the world. However, sometimes we want a body to collide only with a given
group of bodies and not with other bodies. This is called collision filtering. The idea is to group the collision shapes of bodies into categories. Then we can specify
for each collision shape against which categories it will be able to collide. \\
ReactPhysics3D uses bits mask to represent categories. The first thing to do is to assign a category to the collision shapes of your body. To do this, you need to
2018-03-19 22:07:09 +00:00
call the \texttt{ProxyShape::setCollisionCategoryBits()} method on the corresponding proxy shape as in the following example. Here we consider that we have four bodies
2015-03-01 21:37:22 +00:00
where each one has a single collision shape. \\
\begin{lstlisting}
// Enumeration for categories
enum Category {
CATEGORY1 = 0x0001,
CATEGORY2 = 0x0002,
CATEGORY3 = 0x0004
};
// Set the collision category for each proxy shape of
// each of the four bodies
proxyShapeBody1->setCollisionCategoryBits(CATEGORY1);
proxyShapeBody2->setCollisionCategoryBits(CATEGORY2);
proxyShapeBody3->setCollisionCategoryBits(CATEGORY3);
proxyShapeBody4->setCollisionCategoryBits(CATEGORY3);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
2015-03-01 21:37:22 +00:00
\vspace{0.6cm}
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
As you can see, the collision shape of body 1 will be part of the category 1, the collision shape of body 2 will be part of the category 2 and the collision shapes of bodies 3 and 4 will be
part of the category 3. \\
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
\begin{sloppypar}
Now, for each collision shape, we need to specify with which categories the shape is allowed to collide with. To do this, you need to use the \texttt{ProxyShape::setCollideWithMaskBits()}
2018-03-19 22:07:09 +00:00
method of the proxy shape. Note that you can specify one or more categories using the bitwise OR operator. The following example shows how to specify with which categories the
2015-03-01 21:37:22 +00:00
shapes can collide. \\
\end{sloppypar}
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// For each shape, we specify with which categories it
// is allowed to collide
proxyShapeBody1->setCollideWithMaskBits(CATEGORY3);
proxyShapeBody2->setCollideWithMaskBits(CATEGORY1 | CATEGORY3);
proxyShapeBody3->setCollideWithMaskBits(CATEGORY2);
proxyShapeBody4->setCollideWithMaskBits(CATEGORY2);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
2015-03-01 21:37:22 +00:00
\vspace{0.6cm}
As you can see, we specify that the body 1 will be allowed to collide with bodies from the categorie 3. We also indicate that the body 2 will be allowed to collide with bodies from the
category 1 and 3 (using the bitwise OR operator). Finally, we specify that bodies 3 and 4 will be allowed to collide against bodies of the category 2. \\
A collision shape is able to collide with another only if you have specify that the category mask of the first shape is part of the \emph{collide with} mask of the second shape. It
is also important to understand that this condition must be satisfied in both directions. For instance in the previous example, the body 1 (of category 1) says that it wants to collide
against bodies of the category 3 (for instance against body 3). However, body 1 and body 3 will not be able to collide because the body 3 does not say that it wants to collide
with bodies from category 1. Therefore, in the previous example, the body 2 is allowed to collide against bodies 3 and 4 but no other collision is allowed. \\
In the same way, you can perform this filtering for ray casting (described in section \ref{sec:raycasting}). For instance, you can perform a ray cast test
against a given subset of categories of collision shapes only.
2013-09-24 22:02:09 +00:00
\section{Joints}
2018-03-19 22:07:09 +00:00
Joints are used to constrain the motion of the rigid bodies between each other. A single joint represents a constraint between two rigid bodies.
2013-09-24 22:02:09 +00:00
When the motion of the first body of the joint is known, the relative motion of the second body has at most six degrees of freedom (three for the
translation and three for the rotation). The different joints can reduce the number of degrees of freedom between two rigid bodies. \\
2013-10-06 16:46:32 +00:00
Some joints have limits to control the range of motion and some joints have motors to automatically move the bodies of the joint at a given speed. \\
2016-04-09 20:52:25 +00:00
2013-09-24 22:02:09 +00:00
\subsection{Ball and Socket Joint}
The \texttt{BallAndSocketJoint} class describes a ball and socket joint between two bodies. In a ball and socket joint, the two bodies cannot translate with respect to each other.
2013-10-06 16:46:32 +00:00
However, they can rotate freely around a common anchor point. This joint has three degrees of freedom and can be used to simulate a chain of bodies for instance. \\
2013-09-24 22:02:09 +00:00
2013-10-06 16:46:32 +00:00
In order to create a ball and socket joint, you first need to create an instance of the \texttt{BallAndSocketJointInfo} class with the necessary information. You need to provide the pointers to the
2013-09-24 22:02:09 +00:00
two rigid bodies and also the coordinates of the anchor point (in world-space). At the joint creation, the world-space anchor point will be converted into the local-space of the two rigid
bodies and then, the joint will make sure that the two local-space anchor points match in world-space. Therefore, the two bodies need to be in a correct position at the joint creation. \\
2015-03-01 21:37:22 +00:00
Here is the code to create the \texttt{BallAndSocketJointInfo} object: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Anchor point in world-space
const rp3d::Vector3 anchorPoint(2.0, 4.0, 0.0);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::BallAndSocketJointInfo jointInfo(body1, body2, anchorPoint);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
Now, it is time to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method.
Note that this method will also return a pointer to the \texttt{BallAndSocketJoint} object that has been created internally. You will then
be able to use that pointer to change properties of the joint and also to destroy it at the end. \\
2015-03-01 21:37:22 +00:00
Here is how to create the joint in the world: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the joint in the dynamics world
rp3d::BallAndSocketJoint* joint;
joint = dynamic_cast<rp3d::BallAndSocketJoint*>(world.createJoint(jointInfo));
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
\subsection{Hinge Joint}
2015-03-01 21:37:22 +00:00
The \texttt{HingeJoint} class describes a hinge joint (or revolute joint) between two rigid bodies. The hinge joint only allows rotation around an anchor point and
2013-09-24 22:02:09 +00:00
around a single axis (the hinge axis). This joint can be used to simulate doors or pendulums for instance. \\
In order to create a hinge joint, you first need to create a \texttt{HingeJointInfo} object with the necessary information. You need to provide the pointers to the
two rigid bodies, the coordinates of the anchor point (in world-space) and also the hinge rotation axis (in world-space). The two bodies need to be in a correct position
when the joint is created. \\
2015-03-01 21:37:22 +00:00
Here is the code to create the \texttt{HingeJointInfo} object: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Anchor point in world-space
const rp3d::Vector3 anchorPoint(2.0, 4.0, 0.0);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Hinge rotation axis in world-space
const rp3d::Vector3 axis(0.0, 0.0, 1.0);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
Now, it is time to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method.
Note that this method will also return a pointer to the \texttt{HingeJoint} object that has been created internally. You will then
be able to use that pointer to change properties of the joint and also to destroy it at the end. \\
2015-03-01 21:37:22 +00:00
Here is how to create the joint in the world: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the hinge joint in the dynamics world
rp3d::HingeJoint* joint;
joint = dynamic_cast<rp3d::HingeJoint*>(world.createJoint(jointInfo));
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\subsubsection{Limits}
2015-03-01 21:37:22 +00:00
With the hinge joint, you can constrain the motion range using limits. The limits of the hinge joint are the minimum and maximum angle of rotation allowed with respect to the initial
2013-09-24 22:02:09 +00:00
angle between the bodies when the joint is created. The limits are disabled by default. If you want to use the limits, you first need to enable them by setting the
\texttt{isLimitEnabled} variable of the \texttt{HingeJointInfo} object to \emph{true} before you create the joint. You also have to specify the minimum and maximum limit
angles (in radians) using the \texttt{minAngleLimit} and \texttt{maxAngleLimit} variables of the joint info object. Note that the minimum limit angle must be in the
range $[ -2 \pi; 0 ]$ and the maximum limit angle must be in the range $[ 0; 2 \pi ]$. \\
2015-03-01 21:37:22 +00:00
For instance, here is the way to use the limits for a hinge joint when the joint is created: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Enable the limits of the joint
jointInfo.isLimitEnabled = true;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Minimum limit angle
jointInfo.minAngleLimit = -PI / 2.0;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Maximum limit angle
jointInfo.maxAngleLimit = PI / 2.0;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Create the hinge joint in the dynamics world
rp3d::HingeJoint* joint;
joint = dynamic_cast<rp3d::HingeJoint*>(world.createJoint(jointInfo));
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
2013-09-30 21:55:11 +00:00
\begin{sloppypar}
It is also possible to use the \texttt{HingeJoint::enableLimit()}, \texttt{HingeJoint::setMinAngleLimit()} and \texttt{HingeJoint::setMaxAngleLimit()} methods to specify
the limits of the joint after its creation. See the API documentation for more information.
\end{sloppypar}
2013-09-24 22:02:09 +00:00
\subsubsection{Motor}
2013-10-06 16:46:32 +00:00
A motor is also available for the hinge joint. It can be used to rotate the bodies around the hinge axis at a given angular speed and such that the torque applied to
2013-09-30 21:55:11 +00:00
rotate the bodies does not exceed a maximum allowed torque. The motor is disabled by default. If you want to use it, you first have to activate it using the
2013-09-24 22:02:09 +00:00
\texttt{isMotorEnabled} boolean variable of the \texttt{HingeJointInfo} object before you create the joint. Then, you need to specify the angular motor speed (in radians/seconds)
using the \texttt{motorSpeed} variable and also the maximum allowed torque (in Newton $\cdot$ meters) with the \texttt{maxMotorTorque} variable. \\
2015-03-01 21:37:22 +00:00
For instance, here is how to enable the motor of the hinge joint when the joint is created: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Enable the motor of the joint
jointInfo.isMotorEnabled = true;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Motor angular speed
jointInfo.motorSpeed = PI / 4.0;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Maximum allowed torque
jointInfo.maxMotorTorque = 10.0;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Create the hinge joint in the dynamics world
rp3d::HingeJoint* joint;
joint = dynamic_cast<rp3d::HingeJoint*>(world.createJoint(jointInfo));
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
2013-09-30 21:55:11 +00:00
\begin{sloppypar}
It is also possible to use the \texttt{HingeJoint::enableMotor()}, \texttt{HingeJoint::setMotorSpeed()} and \texttt{HingeJoint::setMaxMotorTorque()} methods to
enable the motor of the joint after its creation. See the API documentation for more information.
\end{sloppypar}
2013-09-24 22:02:09 +00:00
\subsection{Slider Joint}
2015-03-01 21:37:22 +00:00
The \texttt{SliderJoint} class describes a slider joint (or prismatic joint) that only allows relative translation along a single direction. It has a single degree of freedom and allows no
2013-09-30 21:55:11 +00:00
relative rotation. In order to create a slider joint, you first need to specify the anchor point (in world-space) and the slider axis direction (in world-space). The constructor of the
2013-10-06 16:46:32 +00:00
\texttt{SliderJointInfo} object needs two pointers to the bodies of the joint, the anchor point and the axis direction. Note that the two bodies have to be in a correct initial position when
2013-09-30 21:55:11 +00:00
the joint is created. \\
2015-03-01 21:37:22 +00:00
You can see in the following code how to specify the information to create a slider joint: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Anchor point in world-space
const rp3d::Vector3 anchorPoint = rp3d::decimal(0.5) * (body2Position + body1Position);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Slider axis in world-space
const rp3d::Vector3 axis = (body2Position - body1Position);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::SliderJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\vspace{0.6cm}
Now, it is possible to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method.
Note that this method will also return a pointer to the \texttt{SliderJoint} object that has been created internally. You will then
be able to use that pointer to change properties of the joint and also to destroy it at the end. \\
2015-03-01 21:37:22 +00:00
Here is how to create the joint in the world: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the slider joint in the dynamics world
rp3d::SliderJoint* joint;
joint = dynamic_cast<rp3d::SliderJoint*>(world.createJoint(jointInfo));
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\subsubsection{Limits}
It is also possible to control the range of the slider joint motion using limits. The limits are disabled by default. In order to use the limits when the joint is created, you first
need to activate them using the \texttt{isLimitEnabled} variable of the \texttt{SliderJointInfo} class. Then, you need to specify the minimum and maximum translation limits
(in meters) using the \texttt{minTranslationLimit} and \texttt{maxTranslation\-Limit} variables. Note that the initial position of the two bodies when the joint is created
corresponds to a translation of zero. Therefore, the minimum limit must be smaller or equal to zero and the maximum limit must be larger or equal to zero. \\
2015-03-01 21:37:22 +00:00
You can see in the following example how to set the limits when the slider joint is created: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::SliderJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Enable the limits of the joint
jointInfo.isLimitEnabled = true;
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Minimum translation limit
jointInfo.minTranslationLimit = -1.7;
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Maximum translation limit
jointInfo.maxTranslationLimit = 1.7;
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Create the hinge joint in the dynamics world
rp3d::SliderJoint* joint;
joint = dynamic_cast<rp3d::SliderJoint*>(world.createJoint(jointInfo));
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\vspace{0.6cm}
2016-04-09 20:52:25 +00:00
2013-09-30 21:55:11 +00:00
\begin{sloppypar}
2015-03-01 21:37:22 +00:00
You can also use the \texttt{SliderJoint::enableLimit()}, \texttt{SliderJoint::\-setMinTranslationLimit()} and \texttt{SliderJoint::setMaxTranslationLimit()} methods to
enable the limits of the joint after its creation. See the API documentation for more information.
2013-09-30 21:55:11 +00:00
\end{sloppypar}
\subsubsection{Motor}
The slider joint also has a motor. You can use it to translate the bodies along the slider axis at a given linear speed and such that the force applied to
move the bodies does not exceed a maximum allowed force. The motor is disabled by default. If you want to use it when the joint is created, you first have to activate it using the
\texttt{isMotorEnabled} boolean variable of the \texttt{SliderJointInfo} object before you create the joint. Then, you need to specify the linear motor speed (in meters/seconds)
using the \texttt{motorSpeed} variable and also the maximum allowed force (in Newtons) with the \texttt{maxMotorForce} variable. \\
2015-03-01 21:37:22 +00:00
For instance, here is how to enable the motor of the slider joint when the joint is created: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::SliderJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Enable the motor of the joint
jointInfo.isMotorEnabled = true;
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Motor linear speed
jointInfo.motorSpeed = 2.0;
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Maximum allowed force
jointInfo.maxMotorForce = 10.0;
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Create the slider joint in the dynamics world
rp3d::SliderJoint* joint;
joint = dynamic_cast<rp3d::SliderJoint*>(world.createJoint(jointInfo));
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\vspace{0.6cm}
\begin{sloppypar}
It is also possible to use the \texttt{SliderJoint::enableMotor()}, \texttt{SliderJoint::setMotorSpeed()} and \texttt{SliderJoint::setMaxMotorForce()} methods to enable the
motor of the joint after its creation. See the API documentation for more information.
\end{sloppypar}
2013-09-24 22:02:09 +00:00
\subsection{Fixed Joint}
2015-03-01 21:37:22 +00:00
The \texttt{FixedJoint} class describes a fixed joint between two bodies. In a fixed joint, there is no degree of freedom, the bodies are not allowed to translate
2013-09-30 21:55:11 +00:00
or rotate with respect to each other. In order to create a fixed joint, you simply need to specify an anchor point (in world-space) to create the \texttt{FixedJointInfo}
object. \\
2015-03-01 21:37:22 +00:00
For instance, here is how to create the joint info object for a fixed joint: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Anchor point in world-space
rp3d::Vector3 anchorPoint(2.0, 3.0, 4.0);
2013-09-30 21:55:11 +00:00
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::FixedJointInfo jointInfo1(body1, body2, anchorPoint);
2013-09-30 21:55:11 +00:00
\end{lstlisting}
\vspace{0.6cm}
Now, it is possible to create the actual joint in the dynamics world using the \texttt{DynamicsWorld::createJoint()} method.
Note that this method will also return a pointer to the \texttt{FixedJoint} object that has been created internally. You will then
be able to use that pointer to change properties of the joint and also to destroy it at the end. \\
2015-03-01 21:37:22 +00:00
Here is how to create the joint in the world: \\
2013-09-30 21:55:11 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the fixed joint in the dynamics world
rp3d::FixedJoint* joint;
joint = dynamic_cast<rp3d::FixedJoint*>(world.createJoint(jointInfo));
2013-09-30 21:55:11 +00:00
\end{lstlisting}
2013-09-24 22:02:09 +00:00
\subsection{Collision between the bodies of a Joint}
2018-03-19 22:07:09 +00:00
By default, the two bodies involved in a joint are able to collide with each other. However, it is possible to disable the collision between the two bodies that are part
2013-09-24 22:02:09 +00:00
of the joint. To do it, you simply need to set the variable \texttt{isCollisionEnabled} of the joint info object to \emph{false} when you create the joint. \\
2013-10-06 16:46:32 +00:00
For instance, when you create a \texttt{HingeJointInfo} object in order to construct a hinge joint, you can disable the collision between the two bodies of the joint as in the
2015-03-01 21:37:22 +00:00
following example: \\
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Create the joint info object
rp3d::HingeJointInfo jointInfo(body1, body2, anchorPoint, axis);
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Disable the collision between the bodies
jointInfo.isCollisionEnabled = false;
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Create the joint in the dynamics world
rp3d::HingeJoint* joint;
joint = dynamic_cast<rp3d::HingeJoint*>(world.createJoint(jointInfo));
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\subsection{Destroying a Joint}
2013-10-06 16:46:32 +00:00
\begin{sloppypar}
2013-09-24 22:02:09 +00:00
In order to destroy a joint, you simply need to call the \texttt{DynamicsWorld::destroyJoint()} method using the pointer to
2015-03-01 21:37:22 +00:00
a previously created joint object as argument as shown in the following code: \\
2013-10-06 16:46:32 +00:00
\end{sloppypar}
2013-09-24 22:02:09 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// rp3d::BallAndSocketJoint* joint is a previously created joint
2013-09-24 22:02:09 +00:00
2015-03-01 21:37:22 +00:00
// Destroy the joint
world.destroyJoint(joint);
2013-09-24 22:02:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
It is important that you destroy all the joints that you have created at the end of the simulation. Also note that destroying a
2015-03-01 21:37:22 +00:00
rigid body involved in a joint will automatically destroy that joint.
\section{Ray casting}
\label{sec:raycasting}
You can use ReactPhysics3D to test intersection between a ray and the bodies of the world you have created. Ray casting can be performed against multiple bodies, a single body or
any proxy shape of a given body. \\
The first thing you need to do is to create a ray using the \texttt{Ray} class of ReactPhysics3D. As you can see in the following example, this is very easy. You
simply need to specify the point where the ray starts and the point where the ray ends (in world-space coordinates). \\
\begin{lstlisting}
// Start and end points of the ray
rp3d::Vector3 startPoint(0.0, 5.0, 1.0);
rp3d::Vector3 endPoint(0.0, 5.0, 30);
// Create the ray
rp3d::Ray ray(startPoint, endPoint);
\end{lstlisting}
\vspace{0.6cm}
Any ray casting test that will be described in the following sections returns a \texttt{RaycastInfo} object in case of intersection with the ray.
This structure contains the following attributes: \\
\begin{description}
\item[worldPoint] Hit point in world-space coordinates
\item[worldNormal] Surface normal of the proxy shape at the hit point in world-space coordinates
\item[hitFraction] Fraction distance of the hit point between \emph{startPoint} and \emph{endPoint} of the ray. The hit point \emph{p} is such that
$p = startPoint + hitFraction \cdot (endPoint - startPoint)$
2018-03-19 22:07:09 +00:00
\item[body] Pointer to the collision body or rigid body that has been hit by the ray
\item[proxyShape] Pointer to the proxy shape that has been hit by the ray
2015-03-01 21:37:22 +00:00
\end{description}
Note that you can also use collision filtering with ray casting in order to only test ray intersection with specific proxy shapes.
Collision filtering is described in section \ref{sec:collisionfiltering}.
\subsection{Ray casting against multiple bodies}
2018-03-19 22:07:09 +00:00
This section describes how to get all the proxy shapes of all bodies in the world that are intersected by a given ray.
2015-03-01 21:37:22 +00:00
\subsubsection{The RaycastCallback class}
First, you have to implement your own class that inherits from the \texttt{RaycastCallback} class. Then, you need to override the
\texttt{RaycastCallback::notifyRaycastHit()} method in your own class. An instance of your class have to be provided as a parameter
of the raycast method and the \texttt{notifyRaycastHit()} method will be called for each proxy shape that is hit by the ray. You will receive, as a parameter
of this method, a \texttt{RaycastInfo} object that will contain the information about the raycast hit (hit point, hit surface normal, hit body, hit proxy shape, \dots). \\
In your \texttt{notifyRaycastHit()} method, you need to return a fraction value that will specify the continuation of the ray cast after a hit.
The return value is the next maxFraction value to use. If you return a fraction of 0.0, it means that the raycast should terminate. If you return a
fraction of 1.0, it indicates that the ray is not clipped and the ray cast should continue as if no hit occurred. If you return the fraction in the
parameter (hitFraction value in the \texttt{RaycastInfo} object), the current ray will be clipped to this fraction in the next queries. If you return -1.0, it will
ignore this ProxyShape and continue the ray cast. Note that no assumption can be done about the order of the calls of the \texttt{notifyRaycastHit()} method. \\
Here is an example about creating your own raycast callback class that inherits from the \texttt{RaycastCallback} class and how to override the
\texttt{notifyRaycastHit()} method: \\
\begin{lstlisting}
// Class WorldRaycastCallback
class MyCallbackClass : public rp3d::RaycastCallback {
public:
virtual decimal notifyRaycastHit(const RaycastInfo& info) {
// Display the world hit point coordinates
std::cout << "Hit point : " <<
info.worldPoint.x <<
info.worldPoint.y <<
info.worldPoint.z <<
std::endl;
// Return a fraction of 1.0 to gather all hits
return decimal(1.0);
}
};
\end{lstlisting}
\subsubsection{Raycast query in the world}
Now that you have your own raycast callback class, you can use the \texttt{raycast()} method to perform a ray casting test
2018-03-19 22:07:09 +00:00
on a collision world or a dynamics world. \\
2015-03-01 21:37:22 +00:00
The first parameter of this method is a reference to the \texttt{Ray} object representing the ray you need to test intersection with. The second parameter is a pointer to
the object of your raycast callback object. You can specify an optional third parameter which is the bit mask for collision filtering.
It can be used to raycast only against selected categories of proxy shapes as described in section \ref{sec:collisionfiltering}. \\
\begin{lstlisting}
// Create the ray
rp3d::Vector3 startPoint(1 , 2, 10);
rp3d::Vector3 endPoint(1, 2, -20);
Ray ray(startPoint, endPoint);
// Create an instance of your callback class
MyCallbackClass callbackObject;
// Raycast test
world->raycast(ray, &callbackObject);
\end{lstlisting}
\vspace{0.6cm}
\subsection{Ray casting against a single body}
\begin{sloppypar}
2018-03-19 22:07:09 +00:00
You can also perform ray casting against a single specific collision body or rigid body of the world. To do this, you need to use the
2015-03-01 21:37:22 +00:00
\texttt{CollisionBody::raycast()} method. This method takes two parameters. The first one is a reference to the \texttt{Ray} object and the second one
is a reference to the \texttt{RaycastInfo} object that will contain hit information if the ray hits the body. This method returns true if the ray hits the
body. The \texttt{RaycastInfo} object will only be valid if the returned value is \emph{true} (a hit occured). \\
\end{sloppypar}
2016-04-09 20:52:25 +00:00
2015-03-01 21:37:22 +00:00
The following example shows how test ray intersection with a body: \\
\begin{lstlisting}
// Create the ray
rp3d::Vector3 startPoint(1 , 2, 10);
rp3d::Vector3 endPoint(1, 2, -20);
Ray ray(startPoint, endPoint);
// Create the raycast info object for the
// raycast result
RaycastInfo raycastInfo;
// Raycast test
bool isHit = body->raycast(ray, raycastInfo);
\end{lstlisting}
\vspace{0.6cm}
\subsection{Ray casting against the proxy shape of a body}
2018-03-19 22:07:09 +00:00
You can also perform ray casting against a single specific proxy shape of a collision body or rigid body of the world. To do this, you need to use the
\texttt{ProxyShape::raycast()} method of the given proxy shape. This method takes two parameters. The first one is a reference to the \texttt{Ray}
2015-03-01 21:37:22 +00:00
object and the second one is a reference to the \texttt{RaycastInfo} object that will contain hit information if the ray hits the body. This method returns
true if the ray hits the body. The \texttt{RaycastInfo} object will only be valid if the returned value is \emph{true} (a hit occured). \\
2018-03-19 22:07:09 +00:00
The following example shows how to test ray intersection with a given proxy shape: \\
2015-03-01 21:37:22 +00:00
\begin{lstlisting}
// Create the ray
rp3d::Vector3 startPoint(1 , 2, 10);
rp3d::Vector3 endPoint(1, 2, -20);
Ray ray(startPoint, endPoint);
// Create the raycast info object for the
// raycast result
RaycastInfo raycastInfo;
// Test raycasting against a proxy shape
bool isHit = proxyShape->raycast(ray, raycastInfo);
\end{lstlisting}
\vspace{0.6cm}
2013-09-24 22:02:09 +00:00
2016-04-09 20:52:25 +00:00
\section{Testbed application}
\label{sec:testbed}
2013-03-18 21:14:43 +00:00
2018-03-19 22:07:09 +00:00
TODO : Update screenshot here
2016-04-09 20:52:25 +00:00
\begin{figure}[h]
\centering
\includegraphics{testbed.png}
\label{fig:testbed}
\end{figure}
The testbed application is a graphical interface where you can select and see some demo scenes using the
ReactPhysics3D library. \\
2015-03-04 20:31:54 +00:00
2016-04-09 20:52:25 +00:00
Follow the instructions described in section \ref{sec:building} to
compile the testbed application. Note that OpenGL is required to compile it. \\
2013-09-24 22:02:09 +00:00
2016-04-09 20:52:25 +00:00
The testbed application can be found in the \texttt{testbed/} folder of
the ReactPhysics3D library. Do not hesitate to take a look at the code of the demo scenes to better understand how
to use the library in your application. \\
2013-09-24 22:02:09 +00:00
2016-04-09 20:52:25 +00:00
The following subsections describe the demo scenes that can be found in the testbed application.
2015-03-04 20:31:54 +00:00
2016-04-09 20:52:25 +00:00
\subsection{Cubes Scene}
2013-09-24 22:02:09 +00:00
2016-04-09 20:52:25 +00:00
In this scene, you will see how to create a floor and some cubes using the Box Shape for collision detection. Because of gravity,
the cubes will fall down on the floor. After falling down, the cubes will come to rest and start sleeping (become inactive). In this scene,
the cubes will become red as they get inactive (sleeping).
2013-09-24 22:02:09 +00:00
2018-03-06 18:29:16 +00:00
\subsection{Cubes Stack Scene}
This scene has a dynamics world and a pyramid of cubes.
\subsection{Joints Scene}
2013-09-24 22:02:09 +00:00
2018-03-19 22:07:09 +00:00
In this scene, you will learn how to create different joints (ball and socket, hinge, slider, fixed) into the dynamics world. You can also see how
2013-09-24 22:02:09 +00:00
to set the motor or limits of the joints.
2016-04-09 20:52:25 +00:00
\subsection{Collision Shapes Scene}
In this scene, you will see how to create a floor (using the Box Shape) and some other bodies using the different collision shapes available
2018-03-19 22:07:09 +00:00
in the ReactPhysics3D library like capsules, spheres, boxes and convex meshes. Those bodies will fall down to the floor.
2016-04-09 20:52:25 +00:00
\subsection{Heightfield Scene}
2018-03-19 22:07:09 +00:00
In this scene, you will see how to use the Height field collision shape of the library. Several bodies will fall
2016-04-09 20:52:25 +00:00
down to the height field.
2018-03-06 18:29:16 +00:00
\subsection{Raycast Scene}
2015-03-01 21:37:22 +00:00
2016-04-09 20:52:25 +00:00
In this scene, you will see how to use the ray casting methods of the library. Several rays are thrown against the different collision shapes.
It is possible to switch from a collision shape to another using the spacebar key.
2015-03-04 20:31:54 +00:00
2018-03-06 18:29:16 +00:00
\subsection{Collision Detection Scene}
This scene has a collision world and several collision bodies that can be move around with keyboard keys. This scene shows how to manually compute
collision detection in a collision world.
2016-04-09 20:52:25 +00:00
\subsection{Concave Mesh Scene}
In this scene, you will see how to use the static concave mesh collision shape of the library.
2015-03-01 21:37:22 +00:00
2015-08-12 16:14:40 +00:00
\section{Retrieving contacts}
There are several ways to get the contacts information (contact point, normal, penetration depth, \dots) from the \texttt{DynamicsWorld}. \\
\subsection{Contacts of a given rigid body}
If you are interested to retrieve all the contacts of a single rigid body, you can use the \texttt{RigidBody::getContactManifoldsList()} method. This method will
return a linked list with all the current contact manifolds of the body. A contact manifold can contains several contact points. \\
Here is an example showing how to get the contact points of a given rigid body: \\
\begin{lstlisting}
const ContactManifoldListElement* listElem;
// Get the head of the linked list of contact manifolds of the body
listElem = rigidbody->getContactManifoldsList();
// For each contact manifold of the body
for (; listElem != NULL; listElem = listElem->next) {
ContactManifold* manifold = listElem->contactManifold;
// For each contact point of the manifold
for (int i=0; i<manifold->getNbContactPoints(); i++) {
// Get the contact point
ContactPoint* point = manifold->getContactPoint(i);
// Get the world-space contact point on body 1
Vector3 pos = point->getWorldPointOnBody1();
// Get the world-space contact normal
Vector3 normal = point->getNormal();
}
}
\end{lstlisting}
\vspace{0.6cm}
2018-03-19 22:07:09 +00:00
Note that this technique to retrieve the contacts, if you use it between the \texttt{DynamicsWorld::update()} calls, will only give you the contacts at the end of
2015-08-12 16:14:40 +00:00
each frame. You will probably miss several contacts that have occured in the physics internal sub-steps. In section \ref{sec:receiving_feedback}, you will
2018-03-19 22:07:09 +00:00
see how to get all the contact occuring in the physis sub-steps of the engine.
2016-04-09 20:52:25 +00:00
2015-08-12 16:14:40 +00:00
\subsection{All the contacts of the world}
If you want to retrieve all the contacts of any rigid body in the world, you can use the \texttt{DynamicsWorld::getContactsList()} method. This method will
2018-03-19 22:07:09 +00:00
a \texttt{List} with the all the current contact manifolds of the world. A contact manifold may contain several contact points. \\
2015-08-12 16:14:40 +00:00
The following example shows how to get all the contacts of the world using this method: \\
\begin{lstlisting}
2018-03-26 20:10:39 +00:00
rp3d::List<const rp3d::ContactManifold*> manifolds;
2015-08-12 16:14:40 +00:00
// Get all the contacts of the world
manifolds = dynamicsWorld->getContactsList();
2018-03-26 20:10:39 +00:00
rp3d::List<const rp3d::ContactManifold*>::iterator it;
2016-04-09 20:52:25 +00:00
2018-03-26 20:10:39 +00:00
// For each contact manifold
2015-08-12 16:14:40 +00:00
for (it = manifolds.begin(); it != manifolds.end(); ++it) {
2018-03-26 20:10:39 +00:00
const rp3d::ContactManifold* manifold = *it;
2015-08-12 16:14:40 +00:00
2018-03-26 20:10:39 +00:00
// For each contact point of the manifold
rp3d::ContactPoint* contactPoint = manifold->getContactPoints();
while (contactPoint != nullptr) {
2013-09-30 21:55:11 +00:00
2018-03-26 20:10:39 +00:00
// Retrieve the world contact point and normal
rp3d::Vector3 worldPoint = manifold->getShape1()->getLocalToWorldTransform() * contactPoint->getLocalPointOnShape1();
rp3d::Vector3 worldNormal = contactPoint->getNormal();
2015-08-12 16:14:40 +00:00
2018-03-26 20:10:39 +00:00
// Move to the next contact point
contactPoint = contactPoint->getNext();
}
2015-08-12 16:14:40 +00:00
}
\end{lstlisting}
\vspace{0.6cm}
Note that this technique to retrieve the contacts, if you use it between the \texttt{DynamicsWorld::update()} calls, will only give you the contacts are the end of
each frame. You will probably miss several contacts that have occured in the physics internal sub-steps. In section \ref{sec:receiving_feedback}, you will
2018-03-06 18:29:16 +00:00
see how to get all the contact occuring in the physis sub-steps of the engine.
2015-08-12 16:14:40 +00:00
\section{Receiving Feedback}
2018-03-06 18:29:16 +00:00
2015-08-12 16:14:40 +00:00
\label{sec:receiving_feedback}
2018-03-19 22:07:09 +00:00
Sometimes, you want to receive notifications from the physics engine when a given event occurs. The \texttt{EventListener} class can be used for that purpose. In order to use
2013-10-06 16:46:32 +00:00
it, you need to create a new class that inherits from the \texttt{EventListener} class and overrides some methods that will be called by the ReactPhysics3D library when some events
2015-03-01 21:37:22 +00:00
occur. You also need to register your class in the physics world using the \texttt{DynamicsWorld::setEventListener()} as in the following code: \\
2013-10-06 16:46:32 +00:00
\begin{lstlisting}
2015-03-01 21:37:22 +00:00
// Here, YourEventListener is a class that inherits
// from the EventListener class of reactphysics3d
YourEventListener listener;
2013-10-06 16:46:32 +00:00
2015-03-01 21:37:22 +00:00
// Register your event listener class
world.setEventListener(&listener);
2013-10-06 16:46:32 +00:00
\end{lstlisting}
2013-09-30 21:55:11 +00:00
\subsection{Contacts}
2013-10-06 16:46:32 +00:00
If you want to be notified when two bodies that were separated before become in contact, you need to override the \texttt{EventListener::beginContact()} method in your event
listener class. Then, this method will be called when the two separated bodies becomes in contact. \\
If you receive a notification when a new contact between two bodies is found, you need to override the \texttt{EventListener::newContact()} method in your event listener class. Then, this
method will be called when a new contact is found.
2013-09-24 22:02:09 +00:00
\section{Profiler}
2018-03-26 20:10:39 +00:00
\label{sec:profiler}
2018-03-19 22:07:09 +00:00
2013-10-06 16:46:32 +00:00
If you build the library with the \texttt{PROFILING\_ENABLED} variable enabled (see section \ref{sec:cmakevariables}), a real-time profiler will collect information while the application
is running. Then, at the end of your application, when the destructor of the \texttt{DynamicsWorld} class is called, information about the running time of the library will be displayed in the
2018-03-26 20:10:39 +00:00
standard output. This can be useful to know where time is spent in the different parts of the ReactPhysics3D library in case your application is too slow. \\
2013-10-06 16:46:32 +00:00
2018-03-26 20:10:39 +00:00
Each collision or dynamics world has its own profiler. By default, the profiling report wil be written in a text file next to the executable.
If you have multiple worlds in your application, there will be one profile file for each world. The profile files will be named after the
name of the worlds. By defaults worlds will have names: world, world1, world2, world3, \dots You can change the name of the world by
setting it into the \texttt{WorldSettings} object when you create the world (see section \ref{sec:collisionworld}). \\
It is also possible to output the profiling report to another destination. To do this,
you have to create your own profiler object before the creation of the physics world. You will then be able to add one or more profile destinations
to the profiler. A destination can be either a file or an output stream (\texttt{std::ostream}) of your choice. For each destination, you also
have to select the output format of the profiling report. When this is done, you have to give the pointer to your profiler object in paramater
when you create the world. \\
The following example shows how to create your own profiler object and add a file destination (custom\_profile.txt) and a stream destination (std::cout): \\
\begin{lstlisting}
// Create the profiler
rp3d::Profiler* profiler = new rp3d::Profiler();
// Add a log destination file
profiler->addFileDestination("custom\_profile.txt", Profiler::Format::Text);
2018-03-19 22:07:09 +00:00
2018-03-26 20:10:39 +00:00
// Add an output stream destination
profiler->addStreamDestination(std::cout, Profiler::Format::Text);
// Create the physics world with your profiler
rp3d::CollisionWorld world(rp3d::WorldSettings(), nullptr, profiler);
\end{lstlisting}
\section{Logger}
\label{sec:logger}
2018-03-19 22:07:09 +00:00
ReactPhysics3D has an internal logger that can be used to output logs while running the application. This can be useful for debugging for instance.
2018-03-26 20:10:39 +00:00
To enable the logger, you need to build the library with the \texttt{LOGS\_ENABLED} variable enabled (see section \ref{sec:cmakevariables}). \\
2018-03-19 22:07:09 +00:00
Each collision or dynamics world has its own logger. By default, logs wil be written in an HTML file next to the executable.
If you have multiple worlds in your application, there will be one log file for each world. The logs files will be named after the
name of the worlds. By defaults worlds will have names: world, world1, world2, world3, \dots You can change the name of the world by
setting it into the \texttt{WorldSettings} object when you create the world (see section \ref{sec:collisionworld}). \\
2018-03-26 20:10:39 +00:00
It is also possible to output the logs to another destination. To do this,
you have to create your own logger object before the creation of the physics world. You will then be able to add one or more logs destinations
2018-03-19 22:07:09 +00:00
to the logger. A destination can be either a file or an output stream (\texttt{std::ostream}) of your choice. For each destination, you also
have to select the output format of the logs (text or HTML). When this is done, you have to give the pointer to your logger object in paramater
when you create the world. \\
2018-03-26 20:10:39 +00:00
The following example shows how to create your own logger object and add a file destination (custom\_log.html) and a stream destination (std::cout): \\
2018-03-19 22:07:09 +00:00
\begin{lstlisting}
2018-03-26 20:10:39 +00:00
// Create the logger
rp3d::Logger* logger = new rp3d::Logger();
// Log level (infor, warning and error
uint logLevel = static\_cast<uint>(Logger::Level::Info) | static\_cast<uint>(Logger::Level::Warning) |
static\_cast<uint>(Logger::Level::Error);
// Add a log destination file
logger->addFileDestination("custom\_log.html", logLevel, Logger::Format::HTML);
// Add an output stream destination
logger->addStreamDestination(std::cout, logLevel, Logger::Format::Text);
// Create the physics world with your logger
rp3d::CollisionWorld world(rp3d::WorldSettings(), logger);
2018-03-19 22:07:09 +00:00
\end{lstlisting}
\vspace{0.6cm}
Do not forget to build the ReactPhysics3D library without the logs when you release your final application.
2013-03-18 21:14:43 +00:00
\section{API Documentation}
Some documentation about the API of the code has been generated
using Doxygen. You will be able to find this documentation in the library archive in the folder \texttt{/documentation/API/html/}. You just
need to open the \texttt{index.html} file with your favorite web browser.
2018-03-19 22:07:09 +00:00
\section{Bugs}
If you find some bugs, do not hesitate to report them on our issue tracker here: \\
2013-03-18 21:14:43 +00:00
2018-03-19 22:07:09 +00:00
\url{https://github.com/DanielChappuis/reactphysics3d/issues} \\
2013-03-18 21:14:43 +00:00
2018-03-19 22:07:09 +00:00
Thanks a lot for reporting the issues that you find. It will help us to correct and improve the library.
2016-04-09 20:52:25 +00:00
2013-03-18 21:14:43 +00:00
\end{document}