using System.Collections; using System.Collections.Generic; using UnityEngine; public class AreaAgent : Agent { public GameObject area; public override List CollectState() { List state = new List(); Vector3 velocity = GetComponent().velocity; state.Add((transform.position.x - area.transform.position.x)); state.Add((transform.position.y - area.transform.position.y)); state.Add((transform.position.z + 5 - area.transform.position.z)); state.Add(velocity.x); state.Add(velocity.y); state.Add(velocity.z); return state; } public void MoveAgent(int movement) { float directionX = 0; float directionZ = 0; float directionY = 0; if (movement == 1) { directionX = -1; } if (movement == 2) { directionX = 1; } if (movement == 3) { directionZ = -1; } if (movement == 4) { directionZ = 1; } if (movement == 5 && GetComponent().velocity.y <= 0) { directionY = 1; } float edge = 0.499f; float rayDepth = 0.51f; Vector3 fwd = transform.TransformDirection(Vector3.down); if (!Physics.Raycast(transform.position, fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(edge, 0f, 0f), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(-edge, 0f, 0f), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(0.0f, 0f, edge), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(0.0f, 0f, -edge), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(edge, 0f, edge), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(-edge, 0f, edge), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(edge, 0f, -edge), fwd, rayDepth) && !Physics.Raycast(transform.position + new Vector3(-edge, 0f, -edge), fwd, rayDepth)) { directionY = 0f; directionX = directionX / 5f; directionZ = directionZ / 5f; } gameObject.GetComponent().AddForce(new Vector3(directionX * 40f, directionY * 300f, directionZ * 40f)); if (GetComponent().velocity.sqrMagnitude > 25f) { GetComponent().velocity *= 0.95f; } } public override void AgentStep(float[] act) { reward = -0.01f; int movement = Mathf.FloorToInt(act[0]); MoveAgent(movement); if (gameObject.transform.position.y < 0.0f || Mathf.Abs(gameObject.transform.position.x - area.transform.position.x) > 8f || Mathf.Abs(gameObject.transform.position.z + 5 - area.transform.position.z) > 8) { done = true; reward = -1f; } } public override void AgentReset() { transform.position = new Vector3(Random.Range(-3.5f, 3.5f), 1.1f, -8f) + area.transform.position; GetComponent().velocity = new Vector3(0f, 0f, 0f); area.GetComponent().ResetArea(); } }