us.ihmc.simulationconstructionset.physics.collision.simple.CapsuleShapeDescription Maven / Gradle / Ivy
package us.ihmc.simulationconstructionset.physics.collision.simple;
import us.ihmc.euclid.geometry.BoundingBox3D;
import us.ihmc.euclid.geometry.LineSegment3D;
import us.ihmc.euclid.geometry.interfaces.LineSegment3DReadOnly;
import us.ihmc.euclid.transform.RigidBodyTransform;
import us.ihmc.euclid.tuple3D.Point3D;
import us.ihmc.euclid.tuple3D.Vector3D;
import us.ihmc.euclid.tuple3D.interfaces.Point3DBasics;
import us.ihmc.simulationconstructionset.physics.CollisionShapeDescription;
public class CapsuleShapeDescription> implements CollisionShapeDescription
{
private double radius;
private LineSegment3D lineSegmentInShapeFrame = new LineSegment3D();
private LineSegment3D lineSegment = new LineSegment3D();
private final BoundingBox3D boundingBox = new BoundingBox3D(Double.NEGATIVE_INFINITY,
Double.NEGATIVE_INFINITY,
Double.NEGATIVE_INFINITY,
Double.POSITIVE_INFINITY,
Double.POSITIVE_INFINITY,
Double.POSITIVE_INFINITY);
private boolean boundingBoxNeedsUpdating = true;
private RigidBodyTransform transform = new RigidBodyTransform();
private RigidBodyTransform tempTransform = new RigidBodyTransform();
public CapsuleShapeDescription(double radius, LineSegment3DReadOnly lineSegment)
{
this.radius = radius;
this.lineSegmentInShapeFrame.set(lineSegment);
this.lineSegment.set(lineSegment);
boundingBoxNeedsUpdating = true;
}
public CapsuleShapeDescription(double radius, double height)
{
if (height < 2.0 * radius)
throw new RuntimeException("Capsule height must be at least 2.0 * radius!");
this.radius = radius;
this.lineSegmentInShapeFrame.set(0.0, 0.0, -height / 2.0 + radius, 0.0, 0.0, height / 2.0 - radius);
this.lineSegment.set(0.0, 0.0, -height / 2.0 + radius, 0.0, 0.0, height / 2.0 - radius);
boundingBoxNeedsUpdating = true;
}
@Override
public CapsuleShapeDescription copy()
{
CapsuleShapeDescription copy = new CapsuleShapeDescription<>(radius, lineSegment);
copy.setTransform(this.transform);
return copy;
}
public double getRadius()
{
return radius;
}
public void getTransform(RigidBodyTransform transformToPack)
{
transformToPack.set(transform);
}
public void setTransform(RigidBodyTransform transform)
{
this.transform.set(transform);
this.lineSegment.applyTransform(transform);
}
public void getLineSegment(LineSegment3D lineSegmentToPack)
{
lineSegmentToPack.set(lineSegment);
}
public void getLineSegmentInShapeFrame(LineSegment3D lineSegmentInShapeFrameToPack)
{
lineSegmentInShapeFrameToPack.set(lineSegmentInShapeFrame);
}
@Override
public void setFrom(T capsuleShapeDescription)
{
this.radius = capsuleShapeDescription.getRadius();
capsuleShapeDescription.getLineSegment(this.lineSegment);
capsuleShapeDescription.getLineSegmentInShapeFrame(this.lineSegmentInShapeFrame);
boundingBoxNeedsUpdating = true;
capsuleShapeDescription.getTransform(this.transform);
}
@Override
public void applyTransform(RigidBodyTransform transformToWorld)
{
transform.preMultiply(transformToWorld);
lineSegment.applyTransform(transformToWorld);
boundingBoxNeedsUpdating = true;
}
@Override
public void getBoundingBox(BoundingBox3D boundingBoxToPack)
{
if (boundingBoxNeedsUpdating)
{
updateBoundingBox();
boundingBoxNeedsUpdating = false;
}
boundingBoxToPack.set(boundingBox);
}
private void updateBoundingBox()
{
Point3DBasics firstEndpoint = lineSegment.getFirstEndpoint();
Point3DBasics secondEndpoint = lineSegment.getSecondEndpoint();
double xMin, yMin, zMin, xMax, yMax, zMax;
if (firstEndpoint.getX() < secondEndpoint.getX())
{
xMin = firstEndpoint.getX();
xMax = secondEndpoint.getX();
}
else
{
xMin = secondEndpoint.getX();
xMax = firstEndpoint.getX();
}
if (firstEndpoint.getY() < secondEndpoint.getY())
{
yMin = firstEndpoint.getY();
yMax = secondEndpoint.getY();
}
else
{
yMin = secondEndpoint.getY();
yMax = firstEndpoint.getY();
}
if (firstEndpoint.getZ() < secondEndpoint.getZ())
{
zMin = firstEndpoint.getZ();
zMax = secondEndpoint.getZ();
}
else
{
zMin = secondEndpoint.getZ();
zMax = firstEndpoint.getZ();
}
boundingBox.set(xMin - radius, yMin - radius, zMin - radius, xMax + radius, yMax + radius, zMax + radius);
}
@Override
public boolean isPointInside(Point3D pointInWorld)
{
return (lineSegment.distanceSquared(pointInWorld) <= radius * radius);
}
private final Point3D tempPointForRollingWorldFrame = new Point3D();
private final Point3D tempPointForRollingShapeFrame = new Point3D();
private final Vector3D tempVectorForRollingWorldFrame = new Vector3D();
private final Vector3D tempVectorForRollingShapeFrame = new Vector3D();
// private final Vector3D tempVectorTwoForRollingShapeFrame = new Vector3D();
// private final LineSegment3d tempLineSegmentForRollingWorldFrame = new LineSegment3d();
@Override
public boolean rollContactIfRolling(Vector3D surfaceNormal, Point3D pointToRoll)
{
tempTransform.set(transform);
tempTransform.invert();
// System.out.println("tempTransform = " + tempTransform);
tempPointForRollingWorldFrame.set(pointToRoll);
tempVectorForRollingWorldFrame.set(surfaceNormal);
// tempLineSegmentForRollingWorldFrame.set(lineSegment);
tempTransform.transform(tempPointForRollingWorldFrame);
tempTransform.transform(tempVectorForRollingWorldFrame);
// tempLineSegmentForRollingWorldFrame.applyTransform(tempTransform);
boolean isRolling = rollContactIfRollingShapeFrame(tempVectorForRollingWorldFrame, tempPointForRollingWorldFrame);
transform.transform(tempPointForRollingWorldFrame);
pointToRoll.set(tempPointForRollingWorldFrame);
return isRolling;
}
public boolean rollContactIfRollingShapeFrame(Vector3D surfaceNormalInShapeFrame, Point3D pointToRollInShapeFrame)
{
//TODO: This assumes shape frame has the capsule being along the z axis and middle of capsule at 0, 0...
//TODO: If we don't want to assume that, we should just do everything in world frame then and use the vectors and dot products, etc.
// System.out.println("\nCapsuleRolling:");
// System.out.println("surfaceNormalInShapeFrame = " + surfaceNormalInShapeFrame);
// System.out.println("pointToRollInShapeFrame = " + pointToRollInShapeFrame);
// System.out.println("lineSegmentInShapeFrame = " + lineSegmentInShapeFrame);
tempVectorForRollingShapeFrame.set(surfaceNormalInShapeFrame);
double height = 2.0 * (lineSegmentInShapeFrame.getSecondEndpoint().getZ() + radius);
// System.out.println("pointToRollInShapeFrame.getZ() = " + pointToRollInShapeFrame.getZ());
// System.out.println("height = " + height);
// System.out.println("radius = " + radius);
// System.out.println("-height/2.0 + radius = " + (-height/2.0 + radius));
// System.out.println("height/2.0 - radius = " + (height/2.0 - radius));
if ((pointToRollInShapeFrame.getZ() > -height / 2.0 + radius) && (pointToRollInShapeFrame.getZ() < height / 2.0 - radius))
{
// Not on the caps. So don't roll along the flat side, only along the curved side...
tempVectorForRollingShapeFrame.setZ(0.0);
// System.out.println("Not on the caps... surfaceNormalInShapeFrame = " + tempVectorForRollingShapeFrame);
}
lineSegmentInShapeFrame.orthogonalProjection(pointToRollInShapeFrame, tempPointForRollingShapeFrame);
// System.out.println("after line projection pointToRollInShapeFrame = " + tempPointForRollingShapeFrame);
double currentRadiusOfPenetration = pointToRollInShapeFrame.distance(tempPointForRollingShapeFrame);
tempVectorForRollingShapeFrame.normalize();
tempVectorForRollingShapeFrame.scale(currentRadiusOfPenetration);
// tempVectorForRolling.scale(radius);
tempPointForRollingShapeFrame.add(tempVectorForRollingShapeFrame);
pointToRollInShapeFrame.set(tempPointForRollingShapeFrame);
// System.out.println("pointToRollInShapeFrame = " + pointToRollInShapeFrame);
//TODO: Not necessarily true all the time....
return true;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy