Prevent character move with pickup

In my 3rd person game, I have a player which is being controlled with a Character Controller, with a certain radius and it’s colliding with the world’s walls etc all OK. The problem comes when I try and make the character pickup and drop an object.

I’m detecting the pickup object with isTrigger and then parenting that object inside the player hierarchy so the character appears to carry it. However, the character + pickup is larger than the original radius, so it’s possible to push the carried pickup into a wall. So, in the picked up script I make the radius larger which almost works.

The problems start when the pickup is dropped close to a wall. In this case, when the radius is enlarged, it will overlap any nearby colliders. This situation isn’t detected by the controller.Move function so the player is able to travel through the wall.

I’ve tried testing for overlap with Physics.OverlapSphere and then forcing the player away from the wall but of course the movement glitches.

I’m wondering if anyone has any ideas?

How about increasing the radius only after the character moved away from the wall?
Instead of increasing the radius when you pick up the object, start a coroutine that will check if the character is touching something (Use Physics.OverlapSphere just like you did earlier).
Only after the character moves away from the wall, the coroutine will not find him touching anything, and in turn increase the radius of the character collider…

It leaves a chance that right after the character picks up the object, he can move closer to the wall and the item will enter it, but only once. When it’s away from the wall the collider will increase and all will be good and dandy.

If you want to cancel that chance completely, inside the coroutine (after you already picked up the object) you can add a raycast to the move direction of the character and stop it from moving to that direction.

Thanks! That gave me some more ideas and sometimes that’s all you need :slight_smile:

Here’s my code, maybe someone else will find this useful too.

IEnumerator AdjustRadiusForPickup()
{
	// Enlarge to encompass pickup
	controller.radius = 1.7f;
	controller.center = new Vector3(0, 1.7f, 0);

	bool once = true;

	// 1. Check for overlapping spheres and adjust radius
	while (once  pickup.transform.parent == pusher.transform)
	{
		// Is there an overlap?
		Collider[] cols = Physics.OverlapSphere(transform.position, 1.7f);
		foreach (Collider col in cols)
		{
			if (col.CompareTag("building") || col.CompareTag("wall"))
			{
				// Use default radius to prevent movement thru walls 
				controller.radius = 0.5f;
				controller.center = new Vector3(0, 1, 0);
				once = false;

				// Prevent dropping the pickup while it's in the wall
				canDrop = false;
			}
		}
		yield return new WaitForFixedUpdate();
	}

	// 2. If moved away from the wall enlarge radius
	bool movedAway = false;
	while (movedAway == false  pickup.transform.parent == pusher.transform)
	{
		Collider[] cols = Physics.OverlapSphere(transform.position, 1.7f);
		bool buildingwall = false;
		foreach (Collider col in cols)
		{
			if (col.CompareTag("building") || col.CompareTag("wall"))
			{
				buildingwall = true;
			}
		}
		if (buildingwall == false)
		{
			// No longer inside a building or wall enlarge radius
			movedAway = true;
			controller.radius = 1.7f;
			controller.center = new Vector3(0, 1.7f, 0);
		}
		yield return new WaitForFixedUpdate();
	}

	// Now allowed to drop pickup again
	canDrop = true;

	yield return 0;
}

Or, alternatively, you could give the object a collider and no rigidbody instead. (Or a collider and an isKinematic rigidbody) You’ll still collide with it, you’ll still be able to pick it up, but it’d be prevented from going through the wall and impact things like a true piece of the player.

Yeah I tried that but couldn’t get it to work with the Character Controller