• Edited

Hi,

I'm putting together a football/soccer match engine as a learning exercise and I've run into a problem trying to freeze/attach a the ball to one of the players. Here's what happens:

  1. Players move to their position
  2. Once players are in position one of them takes the kick off and passes the ball to another player behind them
  3. The player is a CharacterBody3d with an Area3D around their feet. The ball enters the area and the idea is that it "sticks" to the player in the exact position it enters the Area3d.
  4. The player should then run forward with the ball which should be frozen at point of entering Area3D.

What actually happens is that the ball gets frozen as expected but as soon as the player moves forward, the ball shifts position - it doesn't stay in front of the player.

I've debugged this, even with a sleep timer and the freeze looks to work correctly. I had half expected the physics to not have been
complete so it shifted but the sleep timer discounts that theory - I think.

To demonstrate the problem here are a couple of images:

  1. The point the ball enters the Area3D and "freezes"

  2. GIF showing the ball "shift" once the player moves

The on body entered method looks like this:

public void OnFeetEntered(Node3D body)
{
    if(CurrentState == PlayerState.KickingOff || CurrentState == PlayerState.MovingToPosition)
    {
        return;
    }

    if (body is Ball ball)
    {
        HasBall = true;
        CurrentState = PlayerState.JustTakenBall;

        ball.LinearVelocity = Vector3.Zero;
        ball.AngularVelocity = Vector3.Zero;
        ball.FreezeMode = RigidBody3D.FreezeModeEnum.Kinematic;
        ball.Freeze = true;

        ball.GlobalPosition = ball.GlobalPosition;
        var localPosition = ToLocal(ball.GlobalPosition);

        AddChild(ball);
        
        ball.Position = localPosition;
    }
}

And this is how the player is told to move (target position is hard-coded to Vector3(-10.0f, 0.252f, -8.0f) for testing purposes):

private void MoveToTarget(Vector3 targetPosition)
{
    Vector3 direction = GlobalPosition.DirectionTo(targetPosition);

    float distance = GlobalPosition.DistanceTo(targetPosition);

    if (distance > stopThreshold)
    {
        Velocity = direction * Speed;

        MoveAndSlide();
    }
    else
    {
        Velocity = Vector3.Zero; // Stop movement

        CurrentState = PlayerState.Idle;

        Look();
    }
}

So what am I doing wrong here? I just want that ball to stay at the position it enters the player when that player moves forward or rotates.

Thanks!

    • Best Answerset by Umble

    The answer to my original question/problem is that the AddChild call was failing because you have to remove the parent first.

    The second issue I ran into was simple recursion, I didn't appreciate that the on entered method would be called repeatedly. This was causing the parent to get removed and added in an infinite loop causing the stack overflow exception. My player is now happily moving forward with the ball in the correct position!

Does the ball need to be a RigidBody3D, which can be moved by the physics engine?

    • Edited

    DaveTheCoder

    I'm still playing around with the concept but yes, I would like it to be moved by physics as it will provide a more interesting match engine I think.

    Someone else pointed out that AddChild requires no parent so I've updated the parent to be removed before the add child.

    That does unfortunately give me another error:

    It doesn't give me any more information so I'm struggling to even know where to look!

    • Best Answerset by Umble

    The answer to my original question/problem is that the AddChild call was failing because you have to remove the parent first.

    The second issue I ran into was simple recursion, I didn't appreciate that the on entered method would be called repeatedly. This was causing the parent to get removed and added in an infinite loop causing the stack overflow exception. My player is now happily moving forward with the ball in the correct position!