iterate through overlapsphere array

Hi all!

I’ve only been using unity or even coding in general for 2 weeks, so sorry if the solution to this is super obvious to you all. This is my first post to the forum, as I am desperate for a solution.

I’m trying to create a targeting system for a hack and slash game using an overlap sphere, but I haven’t been able to find a way to iterate through items in the array. So far the only ‘working’ targeting system only detects and selects the closest enemy (highlighted in green), but I’d like to be able to buttondown and move to the next enemy in the array.

What is the best way to move my through items in an overlapsphere array to reassign my currentTarget?

I’ve tried a few things with limited success… sometimes it jumps between targets, but inconsistently.

Here is my code thus far (be prepared for some hacky garbage):

using System;
using UnityEngine;
public class Targeting : MonoBehaviour
{
    public GameObject currentTarget;
    public GameObject player;
    public bool lockedOn;
    public float sphereRadius;
    public LayerMask layerMask;
    private Vector3 origin;
    private Vector3 direction;
    public Collider[] enemyList;
 
    private void Start()
    {
        lockedOn = false;

    }
    private void Update()
    {
        if (Input.GetButtonDown("Fire2"))
        {
            if(lockedOn== false)
            {
                CollectTargetArray();
               
            }
            else if (lockedOn == true)
            {
                DumpTarget();
            }
        }
    }
    public void CollectTargetArray()
    {
        origin = transform.position + transform.up;
        Collider[] enemyList = Physics.OverlapSphere(origin, sphereRadius, layerMask);
        Array.Sort(enemyList, new DistanceComparer(transform)); //Sorted by Distance
       
        currentTarget = enemyList[i].transform.gameObject;
        player.GetComponent<CharController>().isTargetLocked = true;
        lockedOn = true;
       
        //List all potential targets
        foreach (Collider hit in enemyList)
        {
            Debug.Log(hit.name);
        } 
    }

    void DumpTarget()
    {
        Debug.Log(currentTarget.transform.gameObject.name + " is target dumped.");
        currentTarget = null;
        lockedOn = false;
        player.GetComponent<CharController>().isTargetLocked = false;
    }
    void OnDrawGizmosSelected()
    {
        Gizmos.color = Color.red;
        Debug.DrawLine(origin, origin);
        Gizmos.DrawWireSphere(origin, sphereRadius);
    }
} //END OF SCRIPT

I might be missing it, but where are you getting the variable “i” from that you are using to set your currentTarget?

Also, if the array is already sorted, you should be able to just grab entry 0 to get the nearest enemy, assuming that is how your sort works.

Oops, yeah, “i” was a remnant of something i was using to try and move through the array, but it wasn’t consistently working - sometimes it would grab another target, but then stopped working all together.

Right now the ‘working’ version is grabbing the nearest enemy, but I am looking for a function or mechanism that will let me go through to the next array item on a button press, then return to item 0 after it is through.

Well, I got it working and going through different items on the array. For some reason it didn’t like it when I used a separate function to actually make the change in target and switch all the bools and stuff… but oddly, if I repeated the code in without referring to another function, it seems to work fine.

Here is the working code. Still open to advice on how to clean it up, or an explanation of why I had to repeat the code in all the if statements instead of referring to another function!

public class Targeting : MonoBehaviour
{

    public GameObject currentTarget;
    public GameObject player;

    public bool lockedOn;

    public float sphereRadius;

    public LayerMask layerMask;

    private Vector3 origin;
    private Vector3 direction;

    public Collider[] enemyList;
    public ArrayList[] targetList;
    public int i;
   

    private void Start()
    {
        lockedOn = false;
        i = 0;
    }

    private void LateUpdate()
    {
        if (currentTarget == null)
        {
            lockedOn = false;
            player.GetComponent<CharController>().isTargetLocked = false;
        }

        if (Input.GetButtonDown("Fire2"))
        {
            CollectTargetArray();
        }
        if (Input.GetButtonDown("Fire3"))
        {
            DumpTarget();
        }
    }

    public void CollectTargetArray()
    {
        origin = transform.position + transform.up;
        Collider[] enemyList = Physics.OverlapSphere(origin, sphereRadius, layerMask);
        Array.Sort(enemyList, new DistanceComparer(transform)); //Sorted by Distance

        if (!lockedOn)
        {
            currentTarget = enemyList[i].transform.gameObject;
            player.GetComponent<CharController>().isTargetLocked = true;
            lockedOn = true;
        } else if (lockedOn)
        {
            if (i >= enemyList.Length - 1)
            {
                i = 0;
                currentTarget = enemyList[i].transform.gameObject;
                player.GetComponent<CharController>().isTargetLocked = true;
                lockedOn = true;
            }
            else if (i >= enemyList.Length)
            {
                DumpTarget();
            }
            else
            {
                i++;
                currentTarget = enemyList[i].transform.gameObject;
                player.GetComponent<CharController>().isTargetLocked = true;
                lockedOn = true;
            }
        }


        //List all potential targets and array length
        Debug.Log(enemyList.Length);
        foreach (Collider hit in enemyList)
        {
            Debug.Log(hit.name);
        }   
    }   
    void DumpTarget()
    {
        Debug.Log(currentTarget.transform.gameObject.name + " is target dumped.");
        currentTarget = null;
        lockedOn = false;
        player.GetComponent<CharController>().isTargetLocked = false;
    }

    void OnDrawGizmosSelected()
    {
        Gizmos.color = Color.red;
        Debug.DrawLine(origin, origin);
        Gizmos.DrawWireSphere(origin, sphereRadius);
    }


} //END OF SCRIPT