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

us.ihmc.simulationconstructionset.physics.collision.DefaultCollisionHandler Maven / Gradle / Ivy

There is a newer version: 0.25.3
Show newest version
package us.ihmc.simulationconstructionset.physics.collision;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Random;

import us.ihmc.euclid.tuple3D.Point3D;
import us.ihmc.euclid.tuple3D.Vector3D;
import us.ihmc.simulationconstructionset.ContactingExternalForcePoint;
import us.ihmc.simulationconstructionset.ExternalForcePoint;
import us.ihmc.simulationconstructionset.Link;
import us.ihmc.simulationconstructionset.Robot;
import us.ihmc.simulationconstructionset.physics.CollisionHandler;
import us.ihmc.simulationconstructionset.physics.CollisionShapeWithLink;
import us.ihmc.simulationconstructionset.physics.Contacts;

public class DefaultCollisionHandler implements CollisionHandler
{
   private double velocityForMicrocollision = 0.01; //0.1; //0.1;//0.01;
   private int numberOfCyclesPerContactPair = 1;///4

   private static final boolean DEBUG = false;

   private final Random random;

   private final Vector3D normal = new Vector3D();
   private final Vector3D negative_normal = new Vector3D();

   private Point3D point1 = new Point3D();
   private Point3D point2 = new Point3D();

   private List listeners = new ArrayList<>();

   // Coefficent of restitution.
   private final double epsilon;

   // Coefficient of friction
   private final double mu;

   public DefaultCollisionHandler(double epsilon, double mu)
   {
      this(new Random(), epsilon, mu);
   }

   /**
    * @param epsilon coefficent of restitution.
    * @param mu      coefficient of friction
    * @param robot   Robot model
    */
   public DefaultCollisionHandler(Random random, double epsilon, double mu)
   {
      this.random = random;
      this.epsilon = epsilon;
      this.mu = mu;
   }

   @Override
   public void maintenanceBeforeCollisionDetection()
   {
      shapesInContactList.clear();
   }

   @Override
   public void maintenanceAfterCollisionDetection()
   {
      int numberOfCollisions = shapesInContactList.size();
      if (numberOfCollisions == 0)
         return;

      Collections.shuffle(shapesInContactList, random);

      if (DEBUG)
         System.out.println("Resolving " + numberOfCollisions + " collisions....");
      for (int i = 0; i < numberOfCollisions; i++)
      {
         Contacts shapesInContact = shapesInContactList.get(i);

         //TODO: Get rid of Type cast here...
         CollisionShapeWithLink shape1 = (CollisionShapeWithLink) shapesInContact.getShapeA();
         CollisionShapeWithLink shape2 = (CollisionShapeWithLink) shapesInContact.getShapeB();
         handleLocal(shape1, shape2, shapesInContact);
      }
   }

   private final ArrayList shapesInContactList = new ArrayList<>();

   @Override
   public void handle(Contacts contacts)
   {
      shapesInContactList.add(contacts);
      //      handleLocal(shape1, shape2, contacts);
   }

   private final ArrayList indices = new ArrayList<>();

   private void handleLocal(CollisionShapeWithLink shape1, CollisionShapeWithLink shape2, Contacts contacts)
   {
      boolean shapeOneIsGround = shape1.isGround();
      boolean shapeTwoIsGround = shape2.isGround();

      if (shapeOneIsGround && shapeTwoIsGround)
      {
         // TODO: Make sure ground shapes never get checked for collisions at all...
         //         throw new RuntimeException("Both shapes are ground. Shouldn't be contacting!!");
         return;
      }

      int numberOfContacts = contacts.getNumberOfContacts();
      indices.clear();

      for (int i = 0; i < numberOfContacts; i++)
      {
         indices.add(i);
      }

      // TODO: Smarter way of doing number of cycles.
      // Perhaps prioritize based on velocities or something.
      // Or keep track of graph of collision dependencies...

      for (int cycle = 0; cycle < numberOfCyclesPerContactPair; cycle++)
      {
         // TODO: Sims won't sim same way twice, but I don't think they do anyway...
         Collections.shuffle(indices, random);
         for (int j = 0; j < numberOfContacts; j++)
         {
            int i = indices.get(j);
            double distance = contacts.getDistance(i);

            if (distance > 0.0)
               continue;

            contacts.getWorldA(i, point1);
            contacts.getWorldB(i, point2);

            contacts.getWorldNormal(i, normal);

            if (!contacts.isNormalOnA())
            {
               normal.scale(-1.0);
            }

            // TODO handle the case where the object is embedded inside the object and the normal is invalid
            if (Double.isNaN(normal.getX()))
               throw new RuntimeException("Normal is invalid. Contains NaN!");

            negative_normal.set(normal);
            negative_normal.scale(-1.0);

            Link linkOne = null;
            ExternalForcePoint externalForcePointOne = null;
            Robot robot1 = null;

            Link linkTwo = null;
            ExternalForcePoint externalForcePointTwo = null;
            Robot robot2 = null;

            if (!shapeOneIsGround)
            {
               linkOne = shape1.getLink();
               externalForcePointOne = linkOne.getContactingExternalForcePoints().get(0);
               externalForcePointOne.setOffsetWorld(point1.getX(), point1.getY(), point1.getZ()); // Put the external force points in the right places.
               robot1 = linkOne.getParentJoint().getRobot();
            }

            if (!shapeTwoIsGround)
            {
               linkTwo = shape2.getLink();
               externalForcePointTwo = linkTwo.getContactingExternalForcePoints().get(0);
               externalForcePointTwo.setOffsetWorld(point2.getX(), point2.getY(), point2.getZ());
               robot2 = linkTwo.getParentJoint().getRobot();
            }

            if (!shapeOneIsGround)
            {
               robot1.updateVelocities();
               robot1.update();
            }

            if (!shapeTwoIsGround)
            {
               if (robot2 != robot1)
               {
                  robot2.updateVelocities();
                  robot2.update();
               }
            }

            // Resolve the collision:
            Vector3D p_world = new Vector3D();

            // +++JEP: epsilon, mu hardcoded on construction right now. Need to change that!

            boolean collisionOccurred;

            if (DEBUG)
            {
               System.out.println("shapeOneIsGround = " + shapeOneIsGround);
               System.out.println("shapeTwoIsGround = " + shapeTwoIsGround);
               System.out.println("numberOfContacts = " + numberOfContacts);
               System.out.println("normal = " + normal);
               System.out.println("negative_normal = " + negative_normal);
               System.out.println("point1 = " + point1);
               System.out.println("point2 = " + point2);
               System.out.println("externalForcePointOne = " + externalForcePointOne);
               System.out.println("externalForcePointTwo = " + externalForcePointTwo);
            }

            if (shapeTwoIsGround)
            {
               Vector3D velocityWorld = new Vector3D(0.0, 0.0, 0.0);

               if (externalForcePointOne.getVelocityCopy().lengthSquared() > velocityForMicrocollision * velocityForMicrocollision)
               {
                  collisionOccurred = externalForcePointOne.resolveCollision(velocityWorld, negative_normal, epsilon, mu, p_world); // link1.epsilon, link1.mu, p_world);
               }

               else
               {
                  double penetrationSquared = point1.distanceSquared(point2);
                  externalForcePointOne.resolveMicroCollision(penetrationSquared, velocityWorld, negative_normal, epsilon, mu, p_world);
                  collisionOccurred = true;
               }
            }
            else if (shapeOneIsGround)
            {
               Vector3D velocityWorld = new Vector3D(0.0, 0.0, 0.0);
               if (externalForcePointTwo.getVelocityCopy().lengthSquared() > velocityForMicrocollision * velocityForMicrocollision)
               {
                  collisionOccurred = externalForcePointTwo.resolveCollision(velocityWorld, normal, epsilon, mu, p_world); // link1.epsilon, link1.mu, p_world);
               }

               else
               {
                  double penetrationSquared = point1.distanceSquared(point2);
                  externalForcePointTwo.resolveMicroCollision(penetrationSquared, velocityWorld, normal, epsilon, mu, p_world);
                  collisionOccurred = true;
               }

            }
            else
            {
               Vector3D velocityVectorOne = externalForcePointOne.getVelocityCopy();
               Vector3D velocityVectorTwo = externalForcePointTwo.getVelocityCopy();

               Vector3D velocityDifference = new Vector3D();
               velocityDifference.sub(velocityVectorTwo, velocityVectorOne);

               if (velocityDifference.lengthSquared() > velocityForMicrocollision * velocityForMicrocollision)
               {
                  // Normal Collision
                  collisionOccurred = externalForcePointOne.resolveCollision(externalForcePointTwo, negative_normal, epsilon, mu, p_world); // link1.epsilon, link1.mu, p_world);
               }

               else
               {
                  // MicroCollision
                  double penetrationSquared = point1.distanceSquared(point2);
                  collisionOccurred = externalForcePointOne.resolveMicroCollision(penetrationSquared,
                                                                                  externalForcePointTwo,
                                                                                  negative_normal,
                                                                                  epsilon,
                                                                                  mu,
                                                                                  p_world); // link1.epsilon, link1.mu, p_world);
               }
            }

            if (collisionOccurred)
            {
               for (CollisionHandlerListener listener : listeners)
               {
                  //               System.out.println("collision occured. Visualizing it...");
                  //               System.out.println("externalForcePointOne = " + externalForcePointOne);
                  //               System.out.println("externalForcePointTwo = " + externalForcePointTwo);

                  listener.collision(shape1, shape2, externalForcePointOne, externalForcePointTwo, null, null);
               }
            }
         }
      }
   }

   @Override
   public void addListener(CollisionHandlerListener listener)
   {
      listeners.add(listener);
   }

   @Override
   public void handleCollisions(CollisionDetectionResult results)
   {
      //TODO: Iterate until no collisions left for stacking problems...
      //      for (int j=0; j<10; j++)
      {
         maintenanceBeforeCollisionDetection();

         for (int i = 0; i < results.getNumberOfCollisions(); i++)
         {
            Contacts collision = results.getCollision(i);
            handle(collision);
         }

         maintenanceAfterCollisionDetection();
      }
   }

   @Override
   public void addContactingExternalForcePoints(Link link, List contactingExternalForcePoints)
   {
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy