Ejemplo n.º 1
0
        public static void ResolveCollision(PhysicalBody a, PhysicalBody b, double dt)
        {
            var delta = GetVectorForResolveCollision(a, b);

            if (delta.Length.IsEqual(0))
            {
                return;
            }

            var relativeVelocity = a.Velocity - b.Velocity;

            CorrectPositions(a, b, delta);
            if (relativeVelocity.DotProductWith(delta).IsGreater(0))
            {
                return;
            }
            var normal = delta.SetLength(1);

            double e       = Math.Min(a.Material.elasticity, b.Material.elasticity);
            double impulse = relativeVelocity.DotProductWith(normal) * (1 + e) / (1 / a.Mass + 1 / b.Mass);

            a.Velocity -= impulse * normal / a.Mass;
            b.Velocity += impulse * normal / b.Mass;

            ApplyFrictionForces(a, b, impulse * normal);
        }
Ejemplo n.º 2
0
        public static void ApplyFrictionForces(PhysicalBody a, PhysicalBody b, Point normalImpulse)
        {
            var relativeVelocity = a.Velocity - b.Velocity;
            var ort    = normalImpulse.RotateAroundOrigin(Math.PI / 2);
            var tanget = (relativeVelocity.DotProductWith(ort) * ort);

            if (tanget.Length.IsEqual(0))
            {
                return;
            }
            tanget = tanget.SetLength(1);
            double impulse = -relativeVelocity.DotProductWith(tanget) / (1 / a.Mass + 1 / b.Mass);
            double mu      = Math.Min(a.Material.frictionCoefficient, b.Material.frictionCoefficient);
            Point  frictionImpulse;

            if (Math.Abs(impulse).IsLessOrEqual(normalImpulse.Length * mu))
            {
                frictionImpulse = impulse * tanget;
            }
            else
            {
                frictionImpulse = -(normalImpulse.Length * mu) * tanget;
            }
            a.Velocity += frictionImpulse / a.Mass;
            b.Velocity -= frictionImpulse / b.Mass;
        }
Ejemplo n.º 3
0
        public static void CorrectPositions(PhysicalBody a, PhysicalBody b, Point direction)
        {
            const double percentage  = 1;
            const double slop        = 0.1;
            double       penetration = direction.Length;
            Point        correction  = Math.Max(penetration - slop, 0.0) * percentage /
                                       (1 / a.Mass + 1 / b.Mass) * direction.SetLength(1);

            a.Shape = a.Shape.Move(correction * (1 / a.Mass));
            b.Shape = b.Shape.Move(-correction * (1 / b.Mass));
        }
Ejemplo n.º 4
0
        public bool IsBodyOnGround(PhysicalBody a, Point groundDirection)
        {
            groundDirection = groundDirection.SetLength(1);
            var shape = a.Shape.Move(groundDirection);

            foreach (var body in bodies)
            {
                if (body == a || !IsBodiesCollided(a, body))
                {
                    continue;
                }
                var delta = GetVectorForResolveCollision(a, body);
                if (delta.DotProductWith(groundDirection).IsLess(0))
                {
                    return(true);
                }
            }
            return(false);
        }
Ejemplo n.º 5
0
        public static Point GetVectorForResolveCollision(PhysicalBody a, PhysicalBody b)
        {
            var intersection = a.Shape.IntersectWithPolygon(b.Shape);

            if (intersection.Count < 3)
            {
                return(new Point(0, 0));
            }
            var A             = intersection[0];
            var B             = intersection[1];
            var C             = intersection[intersection.Count - 1];
            var bestDirection = (A.DistanceToPoint(B) < A.DistanceToPoint(C) ? B : C) - A;

            if (a.Shape.GetBoundingBox()[0].DotProductWith(bestDirection).IsGreater(
                    b.Shape.GetBoundingBox()[0].DotProductWith(bestDirection)))
            {
                return(bestDirection);
            }
            return(-bestDirection);
        }
Ejemplo n.º 6
0
 public Collision(PhysicalBody a, PhysicalBody b, double p)
 {
     this.a      = a;
     this.b      = b;
     penetration = p;
 }
Ejemplo n.º 7
0
 public static bool IsBodiesCollided(PhysicalBody a, PhysicalBody b)
 {
     return(a.Shape.IntersectWithPolygon(b.Shape).Count > 2);
 }
Ejemplo n.º 8
0
 public void RemoveBody(PhysicalBody body)
 {
     bodies.Remove(body);
 }
Ejemplo n.º 9
0
 public void AddBody(PhysicalBody body)
 {
     body.Acceleration = acceleration;
     bodies.Add(body);
 }