Aight so strap in this one is pretty damn weird.
Collision.impulse is described as:
…which is a bit fuzzy, but if impulse is obtained by summation, the order of the operations shouldn’t matter, meaning it should be consistent regardless of what order objects are inserted into the physics scene etc. However, I’ve recently been noticing that the impulses are point in opposite directions when the exact same collisions occur based on a few conditions. Below is a gif demonstrating the issue.
oblongleafyblackpanther
I have a small script attached to the box on the right hand side:
using UnityEngine;
public class ImpulseDebugger : MonoBehaviour
{
public Color color = Color.white;
public Vector3 launchDirection = Vector3.forward;
private void Awake()
{
GetComponent<Rigidbody>().velocity = launchDirection;
}
private void OnCollisionEnter(Collision collision)
{
Debug.Log($"GameObject: {gameObject.name}\n" +
$"Contact count: {collision.contactCount}\n" +
$"Impulse: {collision.impulse}");
DrawArrow(collision.contacts[0].point, collision.impulse, 0.5f, color, 1, false);
}
private void DrawArrow(Vector3 position, Vector3 direction, float arrowSize, Color color, float duration = 0, bool depthTest = false)
{
Vector3 cross = Vector3.Cross(direction, Vector3.up).normalized;
if (cross.magnitude < Mathf.Epsilon)
cross = Vector3.Cross(direction, Vector3.right).normalized;
Vector3 upwardsCross = Vector3.Cross(direction, cross).normalized;
Vector3 norm = direction.normalized;
Vector3 tip = position + direction;
Vector3 p1 = tip + cross * arrowSize * 0.7f - norm * arrowSize;
Vector3 p2 = tip - cross * arrowSize * 0.7f - norm * arrowSize;
Vector3 p3 = tip + upwardsCross * arrowSize * 0.7f - norm * arrowSize;
Vector3 p4 = tip - upwardsCross * arrowSize * 0.7f - norm * arrowSize;
Debug.DrawRay(position, direction, color, duration, depthTest);
Debug.DrawRay(tip, p1 - tip, color, duration, depthTest);
Debug.DrawRay(tip, p2 - tip, color, duration, depthTest);
Debug.DrawRay(tip, p3 - tip, color, duration, depthTest);
Debug.DrawRay(tip, p4 - tip, color, duration, depthTest);
}
}
I also have a script that allows me to reload the scene when I press the R key. In the above gif, the first playthrough occurs when I push the play button in the editor, while the second is when I reload the scene with R. We can see the arrow points in opposite directions in each playthrough.
Relevant to this issue is that Unity iterates over the objects in the scene in a different order based off how you load the scene. When you load via pressing play in the Editor, it iterates over objects in an opposite order than when you load the scene dynamically. This mighty be the culprit but why? The collision has the exact same result, shouldn’t the impulse be the exact same?
This issue is making it challenging to perform work in OnCollisionEnter when the impulses are not consistent, since it may also be happening when objects are disabled/enabled. I’ve attached a 2019.4.4f1 project to allow this to be reproduced using the following steps.
- Open the Main scene.
- Duplicate the sphere.
- Delete the old sphere.
- Run the scene.
- Press R to reload it.
- Observe the arrow direction.
The impulse direction will be consistent once the scene is closed/re-opened.
This probably warrants a bug report, but it’s honestly so bizarre I think having a thread is nice. I’ll look into it a bit more and file one when I’m done anyways.
Thanks for any input,
Erik
6119822–666785–Physics-Impulses.zip (21.3 KB)