All Downloads are FREE. Search and download functionalities are using the official Maven repository.

us.ihmc.scs2.simulation.bullet.physicsEngine.BulletTools Maven / Gradle / Ivy

There is a newer version: 17-0.28.1
Show newest version
package us.ihmc.scs2.simulation.bullet.physicsEngine;

import org.bytedeco.bullet.BulletCollision.btBoxShape;
import org.bytedeco.bullet.BulletCollision.btCapsuleShapeZ;
import org.bytedeco.bullet.BulletCollision.btCollisionShape;
import org.bytedeco.bullet.BulletCollision.btCompoundFromGimpactShape;
import org.bytedeco.bullet.BulletCollision.btCompoundShape;
import org.bytedeco.bullet.BulletCollision.btConeShapeZ;
import org.bytedeco.bullet.BulletCollision.btConvexHullShape;
import org.bytedeco.bullet.BulletCollision.btConvexTriangleMeshShape;
import org.bytedeco.bullet.BulletCollision.btCylinderShapeZ;
import org.bytedeco.bullet.BulletCollision.btGImpactMeshShape;
import org.bytedeco.bullet.BulletCollision.btSphereShape;
import org.bytedeco.bullet.BulletCollision.btTriangleMesh;
import org.bytedeco.bullet.LinearMath.btQuaternion;
import org.bytedeco.bullet.LinearMath.btTransform;
import org.bytedeco.bullet.LinearMath.btVector3;

import us.ihmc.euclid.shape.convexPolytope.interfaces.Face3DReadOnly;
import us.ihmc.euclid.shape.convexPolytope.interfaces.Vertex3DReadOnly;
import us.ihmc.euclid.transform.RigidBodyTransform;
import us.ihmc.euclid.transform.interfaces.RigidBodyTransformReadOnly;
import us.ihmc.euclid.tuple3D.Point3D32;
import us.ihmc.euclid.tuple3D.interfaces.Point3DBasics;
import us.ihmc.euclid.tuple3D.interfaces.Tuple3DReadOnly;
import us.ihmc.euclid.tuple3D.interfaces.Vector3DBasics;
import us.ihmc.euclid.tuple4D.Quaternion;
import us.ihmc.log.LogTools;
import us.ihmc.scs2.definition.collision.CollisionShapeDefinition;
import us.ihmc.scs2.definition.geometry.*;

public class BulletTools
{
   public static void toBullet(RigidBodyTransform rigidBodyTransform, btTransform bulletAffineToPack)
   {
      bulletAffineToPack.getOrigin().setValue(rigidBodyTransform.getTranslationX(),
                                              rigidBodyTransform.getTranslationY(),
                                              rigidBodyTransform.getTranslationZ());
      bulletAffineToPack.getBasis().setValue(rigidBodyTransform.getM00(),
                                             rigidBodyTransform.getM01(),
                                             rigidBodyTransform.getM02(),
                                             rigidBodyTransform.getM10(),
                                             rigidBodyTransform.getM11(),
                                             rigidBodyTransform.getM12(),
                                             rigidBodyTransform.getM20(),
                                             rigidBodyTransform.getM21(),
                                             rigidBodyTransform.getM22());
   }

   public static void toEuclid(btTransform bulletAffine, RigidBodyTransform rigidBodyTransform)
   {
      rigidBodyTransform.getRotation().setAndNormalize(bulletAffine.getBasis().getRow(0).getX(),
                                                       bulletAffine.getBasis().getRow(0).getY(),
                                                       bulletAffine.getBasis().getRow(0).getZ(),
                                                       bulletAffine.getBasis().getRow(1).getX(),
                                                       bulletAffine.getBasis().getRow(1).getY(),
                                                       bulletAffine.getBasis().getRow(1).getZ(),
                                                       bulletAffine.getBasis().getRow(2).getX(),
                                                       bulletAffine.getBasis().getRow(2).getY(),
                                                       bulletAffine.getBasis().getRow(2).getZ());
      rigidBodyTransform.getTranslation().setX(bulletAffine.getOrigin().getX());
      rigidBodyTransform.getTranslation().setY(bulletAffine.getOrigin().getY());
      rigidBodyTransform.getTranslation().setZ(bulletAffine.getOrigin().getZ());
   }

   public static void toBullet(Quaternion euclidQuaternion, btQuaternion bulletQuaternion)
   {
      bulletQuaternion.setValue(euclidQuaternion.getX(), euclidQuaternion.getY(), euclidQuaternion.getZ(), euclidQuaternion.getS());
   }

   public static void toEuclid(btQuaternion bulletQuaternion, Quaternion euclidQuaternion)
   {
      euclidQuaternion.set(bulletQuaternion.getX(), bulletQuaternion.getY(), bulletQuaternion.getZ(), bulletQuaternion.getW());
   }

   public static void toBullet(Tuple3DReadOnly euclidTuple, btVector3 bulletVector3)
   {
      bulletVector3.setValue(euclidTuple.getX(), euclidTuple.getY(), euclidTuple.getZ());
   }

   public static void toEuclid(btVector3 bulletVector3, Vector3DBasics euclidVector3D)
   {
      euclidVector3D.set(bulletVector3.getX(), bulletVector3.getY(), bulletVector3.getZ());
   }

   public static void toEuclid(btVector3 bulletVector3, Point3DBasics euclidPoint3D)
   {
      euclidPoint3D.set(bulletVector3.getX(), bulletVector3.getY(), bulletVector3.getZ());
   }

   public static btTriangleMesh convertTriangleMesh3D(RigidBodyTransformReadOnly meshPose, TriangleMesh3DDefinition triangleMesh3DDefinition)
   {
      btTriangleMesh btTriangleMesh = new btTriangleMesh(false, false);

      int[] triangleIndices = triangleMesh3DDefinition.getTriangleIndices();
      Point3D32[] vertices = triangleMesh3DDefinition.getVertices();

      boolean ignorePose = meshPose == null || (!meshPose.hasRotation() && !meshPose.hasTranslation());

      Point3D32 v0 = new Point3D32();
      Point3D32 v1 = new Point3D32();
      Point3D32 v2 = new Point3D32();

      for (int i = 0; i < triangleIndices.length; i += 3)
      {
         if (ignorePose)
         {
            v0 = vertices[triangleIndices[i]];
            v1 = vertices[triangleIndices[i + 1]];
            v2 = vertices[triangleIndices[i + 2]];
         }
         else
         {
            meshPose.transform(vertices[triangleIndices[i]], v0);
            meshPose.transform(vertices[triangleIndices[i + 1]], v1);
            meshPose.transform(vertices[triangleIndices[i + 2]], v2);
         }

         btTriangleMesh.addTriangle(new btVector3(v0.getX(), v0.getY(), v0.getZ()),
                                    new btVector3(v1.getX(), v1.getY(), v1.getZ()),
                                    new btVector3(v2.getX(), v2.getY(), v2.getZ()));
      }

      return btTriangleMesh;
   }

   public static btCollisionShape createBulletCollisionShape(CollisionShapeDefinition collisionShapeDefinition)
   {
      btCollisionShape btCollisionShape = null;

      if (collisionShapeDefinition.getGeometryDefinition() instanceof TriangleMesh3DDefinition triangleMesh3DDefinition)
      {
         btTransform identity = new btTransform();
         btTriangleMesh btTriangleMesh = convertTriangleMesh3D(collisionShapeDefinition.getOriginPose(), triangleMesh3DDefinition);

         if (collisionShapeDefinition.isConcave())
         {
            btGImpactMeshShape btGImpactMeshShape = new btGImpactMeshShape(btTriangleMesh);
            btGImpactMeshShape.updateBound();

            btCompoundFromGimpactShape compoundFromGimpactShape = new btCompoundFromGimpactShape();

            btGImpactMeshShape.setMargin(0.01);
            compoundFromGimpactShape.addChildShape(identity, btGImpactMeshShape);

            btCollisionShape = compoundFromGimpactShape;
         }
         else
         {
            btConvexTriangleMeshShape btConvexTriangleMeshShape = new btConvexTriangleMeshShape(btTriangleMesh);

            btCompoundShape btCompoundShape = new btCompoundShape();

            btConvexTriangleMeshShape.setMargin(0.01);
            btCompoundShape.addChildShape(identity, btConvexTriangleMeshShape);

            btCollisionShape = btCompoundShape;
         }
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Box3DDefinition boxGeometryDefinition)
      {
         btBoxShape boxShape = new btBoxShape(new btVector3(boxGeometryDefinition.getSizeX() / 2.0,
                                                            boxGeometryDefinition.getSizeY() / 2.0,
                                                            boxGeometryDefinition.getSizeZ() / 2.0));
         btCollisionShape = boxShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Sphere3DDefinition sphereGeometryDefinition)
      {
         btSphereShape sphereShape = new btSphereShape(sphereGeometryDefinition.getRadius());
         btCollisionShape = sphereShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Cylinder3DDefinition cylinderGeometryDefinition)
      {
         btCylinderShapeZ cylinderShape = new btCylinderShapeZ(new btVector3(cylinderGeometryDefinition.getRadius(),
                                                                             cylinderGeometryDefinition.getRadius(),
                                                                             cylinderGeometryDefinition.getLength() / 2.0));
         btCollisionShape = cylinderShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Cone3DDefinition coneGeometryDefinition)
      {
         btConeShapeZ coneShape = new btConeShapeZ(coneGeometryDefinition.getRadius(), coneGeometryDefinition.getHeight());
         btCollisionShape = coneShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Capsule3DDefinition capsuleGeometryDefinition)
      {
         if (capsuleGeometryDefinition.getRadiusX() != capsuleGeometryDefinition.getRadiusY()
          || capsuleGeometryDefinition.getRadiusX() != capsuleGeometryDefinition.getRadiusZ()
          || capsuleGeometryDefinition.getRadiusY() != capsuleGeometryDefinition.getRadiusZ())
            LogTools.warn("Bullet capsule does not fully represent the intended capsule!");
         btCapsuleShapeZ capsuleShape = new btCapsuleShapeZ(capsuleGeometryDefinition.getRadiusX(), capsuleGeometryDefinition.getLength());
         btCollisionShape = capsuleShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Ellipsoid3DDefinition ellipsoidGeometryDefinition)
      {
         btSphereShape ellipsoidShape = new btSphereShape(1.0f);
         // Scale the sphere to be an ellipsoid
         btVector3 scaling = new btVector3(ellipsoidGeometryDefinition.getRadiusX(),
                                           ellipsoidGeometryDefinition.getRadiusY(),
                                           ellipsoidGeometryDefinition.getRadiusZ());
         ellipsoidShape.setLocalScaling(scaling);
         btCollisionShape = ellipsoidShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof Ramp3DDefinition rampGeometryDefinition)
      {
         btConvexHullShape convexHullShape = new btConvexHullShape();
         double rampX = rampGeometryDefinition.getSizeX();
         double rampY = rampGeometryDefinition.getSizeY();
         double rampZ = rampGeometryDefinition.getSizeZ();
         // Technically, you only need 6 vertices, but the debug drawing doesn't look as good
         btVector3 v0 = new btVector3(-rampX / 2.0, -rampY / 2.0, 0.0);
         btVector3 v1 = new btVector3(-rampX / 2.0, rampY / 2.0, 0.0);
         btVector3 v2 = new btVector3(rampX / 2.0, rampY / 2.0, 0.0);
         btVector3 v3 = new btVector3(rampX / 2.0, -rampY / 2.0, 0.0);
         btVector3 v4 = new btVector3(rampX / 2.0, rampY / 2.0, rampZ);
         btVector3 v5 = new btVector3(rampX / 2.0, -rampY / 2.0, rampZ);
         convexHullShape.addPoint(v0);
         convexHullShape.addPoint(v1);
         convexHullShape.addPoint(v2);
         convexHullShape.addPoint(v3);
         convexHullShape.addPoint(v0);
         convexHullShape.addPoint(v5);
         convexHullShape.addPoint(v3);
         convexHullShape.addPoint(v5);
         convexHullShape.addPoint(v4);
         convexHullShape.addPoint(v2);
         convexHullShape.addPoint(v4);
         convexHullShape.addPoint(v1);
         btCollisionShape = convexHullShape;
      }
      else if (collisionShapeDefinition.getGeometryDefinition() instanceof ConvexPolytope3DDefinition convexPolytopeDefinition)
      {
         btConvexHullShape convexHullShape = new btConvexHullShape();
         for (Face3DReadOnly face : convexPolytopeDefinition.getConvexPolytope().getFaces())
         {
            for (Vertex3DReadOnly vertex : face.getVertices())
            {
               convexHullShape.addPoint(new btVector3(vertex.getX(), vertex.getY(), vertex.getZ()));
            }
         }
         btCollisionShape = convexHullShape;
      }
      else
      {
         throw new UnsupportedOperationException("Unsupported shape: " + collisionShapeDefinition.getGeometryDefinition().getClass().getSimpleName());
      }

      // For some shapes the default margin seems to be larger,
      // so let's set them all to 1 mm on creation
      btCollisionShape.setMargin(0.001);

      return btCollisionShape;
   }

   public enum eFeatherstoneJointType
   {
      eRevolute, ePrismatic, eSpherical, ePlanar, eFixed, eInvalid
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy