-
Notifications
You must be signed in to change notification settings - Fork 8
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Using distance3d for 2d collision detection and response #87
Comments
Hi @elle-trudgett, distance3d is designed exclusively for 3D! It was never tested for 2D. Having said that, I think the approach that you have should work. When I change the two squares to square_1 = colliders.ConvexHullVertices(array([
(0.0, 0.0, -1000.0),
(0.0, 1.0, -1000.0),
(1.0, 1.0, -1000.0),
(1.0, 0.0, -1000.0),
(0.0, 0.0, 1000.0),
(0.0, 1.0, 1000.0),
(1.0, 1.0, 1000.0),
(1.0, 0.0, 1000.0),
]))
square_2 = colliders.ConvexHullVertices(array([
(0.5, 0.5, -1000.0),
(0.5, 2.0, -1000.0),
(2.0, 2.0, -1000.0),
(2.0, 0.5, -1000.0),
(0.5, 0.5, 1000.0),
(0.5, 2.0, 1000.0),
(2.0, 2.0, 1000.0),
(2.0, 0.5, 1000.0),
])) I get the following result with GJK+EPA:
This makes sense. It is one of the two shortest translation that separates the two squares. However, this case is particularly tricky and you might have discovered a bug here. It is tricky because the other solution is (0.5, 0, 0) and averaging the two solutions is not a good solution. So they are two local and global minima. I would have to take a look how EPA handles these cases.
Yes, EPA doesn't seem to handle flat shapes too well. In a way it is correct, since you have to move an infinitesimal amount along the z axis to separate the two shapes.
I think a more common approach for continuous collision detection is the following (I never tried it though): for a convex mesh of N vertices, you add an additional amount of N vertices that contain the original vertices plus the velocity of the mesh times some delta time. You will get a convex hull around the current and next position of the object. This way you will most likely (not exactly) get an mtv in the opposite direction of the mesh' motion.
It is interesting though that MPR finds the "correct" penetration direction. I would have assumed that it will find something like (0, 0, 1), which easily separates the two flat squares in 3D. I would assume that there are multiple local minima in this case and the selection happens kind of arbitrarily. So I wouldn't rely on this. The trick of extruding the objects in the z dimension should definitely work though.
I think it is a perfect solution. It is exactly in the middle. What's the issue? If it is about the z coordinate, 50 is the middle between 0 and 100. MPR will return 0 if you replace 0 z coordinates by -100 in your square definition. But I guess you don't care about z anyway. |
You need a projection onto u because the mtv might not be what you are looking for. Maybe you are looking for something more similar to the hydroelastic pressure field model that gives you a more smooth direction for a separating force:
I will reopen the issue since you might have discovered some edge case in EPA. I have to investigate it. |
It turns out that the projection is not quite what I'm looking for. So I have this kind of a situation. polyA is moving by vector u to polyA'. There is a box in the way, polyB. Computing the intersection of polyA' with polyB' gives an MTV of (0, -0.25) - I believe - which corresponds to moving the tip of the triangle down to resolve the collision. That might be reasonable in this circumstance, but there are also cases where u is sufficiently large that the MTV would put polyA on the other side of polyB, which should not be possible (it means polyA has moved through polyB.) What I'm trying to calculate, and maybe this isn't the best way of going about it, is the vector u', which goes from the colliding vertex to the intersection point and here polyA can be translated by without overlapping polyB. This is different from projecting the MTV it onto u, shown by the point labeled "Projection", guided by the pink line which is a line perpendicular to u. note that in the case where polyA is larger than polyB it may be the other way around: a vertex of polyB colliding into an edge of polyA. I feel like I don't have quite the right knack for geometry problems to see why I'm making this more complicated than it needs to be. |
The problem is that we have to discretize time in a simulation. The simplest solution for this problem is (similar to the solution that you posted below) to break Another thing that you could take a look at, if you want to spend more time on this, is a version of GJK that computes the intersection of a ray and an convex polytope. There is one in the Jolt physics engine that I didn't translate to Python yet, but should be fairly easy to do. With this version of GJK you could intersect edit: Now that I think about it, I think you could turn each point of polyA plus the corresponding one of polyA' into a polygon and intersect it with polyB. The contact point would be the one that you are looking for. |
I have been using the convex hull of polyA+polyA' to do intersections with polyB, but the contact point, again without knowledge of the movement vector, is not guaranteed to be correct, since it would be the same result as if the "swept polygon" was static. |
I have had some success with taking the minimum distance from all points to all edges (from A to B and vice versa using -u) using line segment intersections (since the distance has to be parallel to u) but it seems very inefficient |
Hi, can
distance3d
work for 2D collision detection and response? I'm attempting to integrate this into a video game. To that end, I'm trying to figure out how far a character collider can move before it collides with some wall/etc.But I'm trying it out for a very simple case and getting unexpected results. Here's a simple example showing 2 squares that overlap and what the minimum translation vector should be (or something similar) shown as an arrow.
I set up the squares like so:
Then I use GJK + EPA to compute the collision and minimum translation vector:
The results that I get are:
My interpretation of this (correct me if I'm wrong) is that the polygons are intersecting (
dist = 0
) but there is no translation needed (mtv = (0, 0, 0)
) which implies that they are touching?I'm not sure how to interpret this result.
The other question I had is that the MTV is not guaranteed to be in the opposite direction of the character's motion (indeed, the GJK/EPA algorithms know nothing of this vector) so it should be as simple as either:
a) Projecting the MTA onto the movement vector, or
b) Normalizing the movement vector and scaling it to the distance to the object
Would that work?
The text was updated successfully, but these errors were encountered: