Search icon CANCEL
Subscription
0
Cart icon
Your Cart (0 item)
Close icon
You have no products in your basket yet
Save more on your purchases! discount-offer-chevron-icon
Savings automatically calculated. No voucher code required.
Arrow left icon
Explore Products
Best Sellers
New Releases
Books
Videos
Audiobooks
Learning Hub
Newsletter Hub
Free Learning
Arrow right icon
timer SALE ENDS IN
0 Days
:
00 Hours
:
00 Minutes
:
00 Seconds
Unity 5.x Game AI Programming Cookbook
Unity 5.x Game AI Programming Cookbook

Unity 5.x Game AI Programming Cookbook: Build and customize a wide range of powerful Unity AI systems with over 70 hands-on recipes and techniques

Arrow left icon
Profile Icon Jorge Palacios Profile Icon Jorge Elieser P Garrido
Arrow right icon
S$66.99
Full star icon Full star icon Half star icon Empty star icon Empty star icon 2.8 (4 Ratings)
Paperback Feb 2025 278 pages 1st Edition
eBook
S$46.99 S$52.99
Paperback
S$66.99
Hardcover
S$66.99
Subscription
Free Trial
Arrow left icon
Profile Icon Jorge Palacios Profile Icon Jorge Elieser P Garrido
Arrow right icon
S$66.99
Full star icon Full star icon Half star icon Empty star icon Empty star icon 2.8 (4 Ratings)
Paperback Feb 2025 278 pages 1st Edition
eBook
S$46.99 S$52.99
Paperback
S$66.99
Hardcover
S$66.99
Subscription
Free Trial
eBook
S$46.99 S$52.99
Paperback
S$66.99
Hardcover
S$66.99
Subscription
Free Trial

What do you get with Print?

Product feature icon Instant access to your digital copy whilst your Print order is Shipped
Product feature icon Paperback book shipped to your preferred address
Product feature icon Redeem a companion digital copy on all Print orders
Product feature icon Access this title in our online reader with advanced features
Product feature icon DRM FREE - Read whenever, wherever and however you want
OR
Modal Close icon
Payment Processing...
tick Completed

Shipping Address

Billing Address

Shipping Methods
Table of content icon View table of contents Preview book icon Preview Book

Unity 5.x Game AI Programming Cookbook

Chapter 1. Behaviors – Intelligent Movement

In this chapter, we will develop AI algorithms for movement by covering the following recipes:

  • Creating the behaviors' template
  • Pursuing and evading
  • Arriving and leaving
  • Facing objects
  • Wandering around
  • Following a path
  • Avoiding agents
  • Avoiding walls
  • Blending behaviors by weight
  • Blending behaviors by priority
  • Combining behaviors using a steering pipeline
  • Shooting a projectile
  • Predicting a projectile's landing spot
  • Targeting a projectile
  • Creating a jump system

Introduction

Unity has been one of the most popular game engines for quite a while now, and it's probably the de facto game development tool for indie developers, not only because of its business model, which has a low entry barrier, but also because of its robust project editor, year-by-year technological improvement, and most importantly, ease of use and an ever-growing community of developers around the globe.

Thanks to Unity's heavy lifting behind the scenes (rendering, physics, integration, and cross-platform deployment, just to name a few) it's possible for us to focus on creating the AI systems that will bring to life our games, creating great real-time experiences in the blink of an eye.

The goal of this book is to give you the tools to build great AI, for creating better enemies, polishing that final boss, or even building your own customized AI engine.

In this chapter, we will start by exploring some of the most interesting movement algorithms based on the steering behavior principles developed by Craig Reynolds, along with work from Ian Millington. These recipes are the stepping stones for most of the AI used in advanced games and other algorithms that rely on movement, such as the family of path-finding algorithms.

Creating the behavior template

Before creating our behaviors, we need to code the stepping stones that help us not only to create only intelligent movement, but also to build a modular system to change and add these behaviors. We will create custom data types and base classes for most of the algorithms covered in this chapter.

Getting ready

Our first step is to remember the update function order of execution:

  • Update
  • LateUpdate

Also, it's important to refresh so that we can select the scripts' order of execution. For our behaviors to work as intended, the rules for ordering are as follows:

  • Agent scripts
  • Behavior scripts
  • Behaviors or scripts based on the previous ones
Getting ready

This is an example of how to arrange the order of execution for the movement scripts. We need to pursue derives from Seek, which derives from AgentBehaviour.

How to do it...

We need to create three classes: Steering, AgentBehaviour, and Agent:

  1. Steering serves as a custom data type for storing the movement and rotation of the agent:
    using UnityEngine;
    using System.Collections;
    public class Steering
    {
        public float angular;
        public Vector3 linear;
        public Steering ()
        {
            angular = 0.0f;
            linear = new Vector3();
        }
    }
  2. Create the AgentBehaviour class, which is the template class for most of the behaviors covered in this chapter:
    using UnityEngine;
    using System.Collections;
    public class AgentBehaviour : MonoBehaviour
    {
        public GameObject target;
        protected Agent agent;
        public virtual void Awake ()
        {
            agent = gameObject.GetComponent<Agent>();
        }
        public virtual void Update ()
        {
                agent.SetSteering(GetSteering());
        }
        public virtual Steering GetSteering ()
        {
            return new Steering();
        }
    }
  3. Finally, Agent is the main component, and it makes use of behaviors in order to create intelligent movement. Create the file and its barebones:
    using UnityEngine;
    using System.Collections;
    public class Agent : MonoBehaviour
    {
        public float maxSpeed;
        public float maxAccel;
        public float orientation;
        public float rotation;
        public Vector3 velocity;
        protected Steering steering;
        void Start ()
        {
            velocity = Vector3.zero;
            steering = new Steering();
        }
        public void SetSteering (Steering steering)
        {
            this.steering = steering;
        }
    }
  4. Next, we code the Update function, which handles the movement according to the current value:
    public virtual void Update ()
    {
        Vector3 displacement = velocity * Time.deltaTime;
        orientation += rotation * Time.deltaTime;
        // we need to limit the orientation values
        // to be in the range (0 – 360)
        if (orientation < 0.0f)
            orientation += 360.0f;
        else if (orientation > 360.0f)
            orientation -= 360.0f;
        transform.Translate(displacement, Space.World);
        transform.rotation = new Quaternion();
        transform.Rotate(Vector3.up, orientation);
    }
  5. Finally, we implement the LateUpdate function, which takes care of updating the steering for the next frame according to the current frame's calculations:
    public virtual void LateUpdate ()
    {
        velocity += steering.linear * Time.deltaTime;
        rotation += steering.angular * Time.deltaTime;
        if (velocity.magnitude > maxSpeed)
        {
            velocity.Normalize();
            velocity = velocity * maxSpeed;
        }
        if (steering.angular == 0.0f)
        {
            rotation = 0.0f;
        }
        if (steering.linear.sqrMagnitude == 0.0f)
        {
            velocity = Vector3.zero;
        }
        steering = new Steering();
    }

How it works...

The idea is to be able to delegate the movement's logic inside the GetSteering() function on the behaviors that we will later build, simplifying our agent's class to a main calculation based on those.

Besides, we are guaranteed to set the agent's steering value before it is used thanks to Unity script and function execution orders.

There's more...

This is a component-based approach, which means that we have to remember to always have an Agent script attached to GameObject for the behaviors to work as expected.

See also

For further information on Unity's game loop and the execution order of functions and scripts, please refer to the official documentation available online at:

Pursuing and evading

Pursuing and evading are great behaviors to start with because they rely on the most basic behaviors and extend their functionality by predicting the target's next step.

Getting ready

We need a couple of basic behaviors called Seek and Flee; place them right after the Agent class in the scripts' execution order.

The following is the code for the Seek behaviour:

using UnityEngine;
using System.Collections;
public class Seek : AgentBehaviour
{
    public override Steering GetSteering()
    {
        Steering steering = new Steering();
        steering.linear = target.transform.position - transform.position;
        steering.linear.Normalize();
        steering.linear = steering.linear * agent.maxAccel;
        return steering;
    }
}

Also, we need to implement the Flee behavior:

using UnityEngine;
using System.Collections;
public class Flee : AgentBehaviour
{
    public override Steering GetSteering()
    {
        Steering steering = new Steering();
        steering.linear = transform.position - target.transform.position;
        steering.linear.Normalize();
        steering.linear = steering.linear * agent.maxAccel;
        return steering;
    }
}

How to do it...

Pursue and Evade are essentially the same algorithm but differ in terms of the base class they derive from:

  1. Create the Pursue class, derived from Seek, and add the attributes for the prediction:
    using UnityEngine;
    using System.Collections;
    
    public class Pursue : Seek
    {
        public float maxPrediction;
        private GameObject targetAux;
        private Agent targetAgent;
    }
  2. Implement the Awake function in order to set up everything according to the real target:
    public override void Awake()
    {
        base.Awake();
        targetAgent = target.GetComponent<Agent>();
        targetAux = target;
        target = new GameObject();
    }
  3. As well as implement the OnDestroy function, to properly handle the internal object:
    void OnDestroy ()
    {
        Destroy(targetAux);
    }
  4. Finally, implement the GetSteering function:
    public override Steering GetSteering()
    {
        Vector3 direction = targetAux.transform.position - transform.position;
        float distance = direction.magnitude;
        float speed = agent.velocity.magnitude;
        float prediction;
        if (speed <= distance / maxPrediction)
            prediction = maxPrediction;
        else
            prediction = distance / speed;
        target.transform.position = targetAux.transform.position;
        target.transform.position += targetAgent.velocity * prediction;
        return base.GetSteering();
    }
  5. To create the Evade behavior, the procedure is just the same, but it takes into account that Flee is the parent class:
    public class Evade : Flee
    {
        // everything stays the same
    }

How it works...

These behaviors rely on Seek and Flee and take into consideration the target's velocity in order to predict where it will go next; they aim at that position using an internal extra object.

Arriving and leaving

Similar to Seek and Flee, the idea behind these algorithms is to apply the same principles and extend the functionality to a point where the agent stops automatically after a condition is met, either being close to its destination (arrive), or far enough from a dangerous point (leave).

Getting ready

We need to create one file for each of the algorithms, Arrive and Leave, respectively, and remember to set their custom execution order.

How to do it...

They use the same approach, but in terms of implementation, the name of the member variables change as well as some computations in the first half of the GetSteering function:

  1. First, implement the Arrive behaviour with its member variables to define the radius for stopping (target) and slowing down:
    using UnityEngine;
    using System.Collections;
    
    public class Arrive : AgentBehaviour
    {
        public float targetRadius;
        public float slowRadius;
        public float timeToTarget = 0.1f;
    }
  2. Create the GetSteering function:
    public override Steering GetSteering()
    {
        // code in next steps
    }
  3. Define the first half of the GetSteering function, in which we compute the desired speed depending on the distance from the target according to the radii variables:
    Steering steering = new Steering();
    Vector3 direction = target.transform.position - transform.position;
    float distance = direction.magnitude;
    float targetSpeed;
    if (distance < targetRadius)
        return steering;
    if (distance > slowRadius)
        targetSpeed = agent.maxSpeed;
    else
        targetSpeed = agent.maxSpeed * distance / slowRadius;
  4. Define the second half of the GetSteering function, in which we set the steering value and clamp it according to the maximum speed:
    Vector3 desiredVelocity = direction;
    desiredVelocity.Normalize();
    desiredVelocity *= targetSpeed;
    steering.linear = desiredVelocity - agent.velocity;
    steering.linear /= timeToTarget;
    if (steering.linear.magnitude > agent.maxAccel)
    {
        steering.linear.Normalize();
        steering.linear *= agent.maxAccel;
    }
    return steering;
  5. To implement Leave, the name of the member variables changes:
    using UnityEngine;
    using System.Collections;
    
    public class Leave : AgentBehaviour
    {
        public float escapeRadius;
        public float dangerRadius;
        public float timeToTarget = 0.1f;
    }
  6. Define the first half of the GetSteering function:
    Steering steering = new Steering();
    Vector3 direction = transform.position - target.transform.position;
    float distance = direction.magnitude;
    if (distance > dangerRadius)
        return steering;
    float reduce;
    if (distance < escapeRadius)
        reduce = 0f;
    else
        reduce = distance / dangerRadius * agent.maxSpeed;
    float targetSpeed = agent.maxSpeed - reduce;
  7. And finally, the second half of GetSteering stays just the same.

How it works...

After calculating the direction to go in, the next calculations are based on two radii distances in order to know when to go full throttle, slow down, and stop; that's why we have several if statements. In the Arrive behavior, when the agent is too far, we aim to full-throttle, progressively slow down when inside the proper radius, and finally to stop when close enough to the target. The converse train of thought applies to Leave.

How it works...

A visual reference for the Arrive and Leave behaviors

Facing objects

Real-world aiming, just like in combat simulators, works a little differently from the widely-used automatic aiming in almost every game. Imagine that you need to implement an agent controlling a tank turret or a humanized sniper; that's when this recipe comes in handy.

Getting ready

We need to make some modifications to our AgentBehaviour class:

  1. Add new member values to limit some of the existing ones:
    public float maxSpeed;
    public float maxAccel;
    public float maxRotation;
    public float maxAngularAccel;
  2. Add a function called MapToRange. This function helps in finding the actual direction of rotation after two orientation values are subtracted:
    public float MapToRange (float rotation) {
        rotation %= 360.0f;
        if (Mathf.Abs(rotation) > 180.0f) {
            if (rotation < 0.0f)
                rotation += 360.0f;
            else
                rotation -= 360.0f;
        }
        return rotation;
    }
  3. Also, we need to create a basic behavior called Align that is the stepping stone for the facing algorithm. It uses the same principle as Arrive, but only in terms of rotation:
    using UnityEngine;
    using System.Collections;
    
    public class Align : AgentBehaviour
    {
        public float targetRadius;
        public float slowRadius;
        public float timeToTarget = 0.1f;
    
        public override Steering GetSteering()
        {
            Steering steering = new Steering();
            float targetOrientation = target.GetComponent<Agent>().orientation;
            float rotation = targetOrientation - agent.orientation;
            rotation = MapToRange(rotation);
            float rotationSize = Mathf.Abs(rotation);
            if (rotationSize < targetRadius)
                return steering;
            float targetRotation;
            if (rotationSize > slowRadius)
                targetRotation = agent.maxRotation;
            else
                targetRotation = agent.maxRotation * rotationSize / slowRadius;
            targetRotation *= rotation / rotationSize;
            steering.angular = targetRotation - agent.rotation;
            steering.angular /= timeToTarget;
            float angularAccel = Mathf.Abs(steering.angular);
            if (angularAccel > agent.maxAngularAccel)
            {
                steering.angular /= angularAccel;
                steering.angular *= agent.maxAngularAccel;
            }
            return steering;
        }
    }

How to do it...

We now proceed to implement our facing algorithm that derives from Align:

  1. Create the Face class along with a private auxiliary target member variable:
    using UnityEngine;
    using System.Collections;
    
    public class Face : Align
    {
        protected GameObject targetAux;
    }
  2. Override the Awake function to set up everything and swap references:
    public override void Awake()
    {
        base.Awake();
        targetAux = target;
        target = new GameObject();
        target.AddComponent<Agent>();
    }
  3. Also, implement the OnDestroy function to handle references and avoid memory issues:
    void OnDestroy ()
    {
        Destroy(target);
    }
  4. Finally, define the GetSteering function:
    public override Steering GetSteering()
    {
        Vector3 direction = targetAux.transform.position - transform.position;
        if (direction.magnitude > 0.0f)
        {
            float targetOrientation = Mathf.Atan2(direction.x, direction.z);
            targetOrientation *= Mathf.Rad2Deg;
            target.GetComponent<Agent>().orientation = targetOrientation;
        }
        return base.GetSteering();
    }

How it works...

The algorithm computes the internal target orientation according to the vector between the agent and the real target. Then, it just delegates the work to its parent class.

Wandering around

This technique works like a charm for random crowd simulations, animals, and almost any kind of NPC that requires random movement when idle.

Getting ready

We need to add another function to our AgentBehaviour class called OriToVec that converts an orientation value to a vector.

public Vector3 GetOriAsVec (float orientation) {
    Vector3 vector  = Vector3.zero;
    vector.x = Mathf.Sin(orientation * Mathf.Deg2Rad) * 1.0f;
    vector.z = Mathf.Cos(orientation * Mathf.Deg2Rad) * 1.0f;
    return vector.normalized;
}

How to do it...

We could see it as a big three-step process in which we manipulate the internal target position in a parameterized random way, face that position, and move accordingly:

  1. Create the Wander class deriving from Face:
    using UnityEngine;
    using System.Collections;
    
    public class Wander : Face
    {
        public float offset;
        public float radius;
        public float rate;
    }
  2. Define the Awake function in order to set up the internal target:
    public override void Awake()
    {
        target = new GameObject();
        target.transform.position = transform.position;
        base.Awake();
    }
  3. Define the GetSteering function:
    public override Steering GetSteering()
    {
        Steering steering = new Steering();
        float wanderOrientation = Random.Range(-1.0f, 1.0f) * rate;
        float targetOrientation = wanderOrientation + agent.orientation;
        Vector3 orientationVec = OriToVec(agent.orientation);
        Vector3 targetPosition = (offset * orientationVec) + transform.position;
        targetPosition = targetPosition + (OriToVec(targetOrientation) * radius);
        targetAux.transform.position = targetPosition;
        steering = base.GetSteering();
        steering.linear = targetAux.transform.position - transform.position;
        steering.linear.Normalize();
        steering.linear *= agent.maxAccel;
        return steering;
    }

How it works...

The behavior takes into consideration two radii in order to get a random position to go to next, looks towards that random point, and converts the computed orientation into a direction vector in order to advance.

How it works...

A visual description of the parameters for creating the Wander behavior

Following a path

There are times when we need scripted routes, and it's just inconceivable to do this entirely by code. Imagine you're working on a stealth game. Would you code a route for every single guard? This technique will help you build a flexible path system for those situations:

Getting ready

We need to define a custom data type called PathSegment:

using UnityEngine;
using System.Collections;

public class PathSegment
{
    public Vector3 a;
    public Vector3 b;

    public PathSegment () : this (Vector3.zero, Vector3.zero){}
    public PathSegment (Vector3 a, Vector3 b)
    {
        this.a = a;
        this.b = b;
    }
}

How to do it...

This is a long recipe that could be seen as a big two-step process. First, we build the Path class, which abstracts points in the path from their specific spatial representations, and then we build the PathFollower behavior, which makes use of that abstraction in order to get actual spatial points to follow:

  1. Create the Path class, which consists of nodes and segments but only the nodes are public and assigned manually:
    using UnityEngine;
    using System.Collections;
    using System.Collections.Generic;
    
    public class Path : MonoBehaviour
    {
        public List<GameObject> nodes;
        List<PathSegment> segments;
    }
  2. Define the Start function to set the segments when the scene starts:
    void Start()
    {
        segments = GetSegments();
    }
  3. Define the GetSegments function to build the segments from the nodes:
    public List<PathSegment> GetSegments ()
    {
        List<PathSegment> segments = new List<PathSegment>();
        int i;
        for (i = 0; i < nodes.Count - 1; i++)
        {
            Vector3 src = nodes[i].transform.position;
            Vector3 dst = nodes[i+1].transform.position;
            PathSegment segment = new PathSegment(src, dst);
            segments.Add(segment);
        }
        return segments;
    }
  4. Define the first function for abstraction, called GetParam:
    public float GetParam(Vector3 position, float lastParam)
    {
        // body
    }
  5. We need to find out which segment the agent is closest to:
    float param = 0f;
    PathSegment currentSegment = null;
    float tempParam = 0f;
    foreach (PathSegment ps in segments)
    {
        tempParam += Vector3.Distance(ps.a, ps.b);
        if (lastParam <= tempParam)
        {
            currentSegment = ps;
            break;
        }
    }
    if (currentSegment == null)
        return 0f;
  6. Given the current position, we need to work out the direction to go to:
    Vector3 currPos = position - currentSegment.a;
    Vector3 segmentDirection = currentSegment.b - currentSegment.a;
    segmentDirection.Normalize();
  7. Find the point in the segment using vector projection:
    Vector3 pointInSegment = Vector3.Project(currPos, segmentDirection);
  8. Finally, GetParam returns the next position to go to along the path:
    param = tempParam - Vector3.Distance(currentSegment.a, currentSegment.b);
    param += pointInSegment.magnitude;
    return param;
  9. Define the GetPosition function:
    public Vector3 GetPosition(float param) 
    {
        // body
    }
  10. Given the current location along the path, we find the corresponding segment:
    Vector3 position = Vector3.zero;
    PathSegment currentSegment = null;
    float tempParam = 0f;
    foreach (PathSegment ps in segments)
    {
        tempParam += Vector3.Distance(ps.a, ps.b);
        if (param <= tempParam)
        {
            currentSegment = ps;
            break;
        }
    }
    if (currentSegment == null)
        return Vector3.zero;
  11. Finally, GetPosition converts the parameter as a spatial point and returns it:
    Vector3 segmentDirection = currentSegment.b - currentSegment.a;
    segmentDirection.Normalize();
    tempParam -= Vector3.Distance(currentSegment.a, currentSegment.b);
    tempParam = param - tempParam;
    position = currentSegment.a + segmentDirection * tempParam;
    return position;
  12. Create the PathFollower behavior, which derives from Seek (remember to set the order of execution):
    using UnityEngine;
    using System.Collections;
    
    public class PathFollower : Seek
    {
        public Path path;
        public float pathOffset = 0.0f;
        float currentParam;
    }
  13. Implement the Awake function to set the target:
    public override void Awake()
    {
        base.Awake();
        target = new GameObject();
        currentParam = 0f;
    }
  14. The final step is to define the GetSteering function, which relies on the abstraction created by the Path class to set the target position and apply Seek:
    public override Steering GetSteering()
    {
        currentParam = path.GetParam(transform.position, currentParam);
        float targetParam = currentParam + pathOffset;
        target.transform.position = path.GetPosition(targetParam);
        return base.GetSteering();
    }

How it works...

We use the Path class in order to have a movement guideline. It is the cornerstone, because it relies on GetParam to map an offset point to follow in its internal guideline, and it also uses GetPosition to convert that referential point to a position in the three-dimensional space along the segments.

The path-following algorithm just makes use of the path's functions in order to get a new position, update the target, and apply the Seek behavior.

There's more...

It's important to take into account the order in which the nodes are linked in the Inspector for the path to work as expected. A practical way to achieve this is to manually name the nodes with a reference number.

There's more...

An example of a path set up in the Inspector window

Also, we could define the OnDrawGizmos function in order to have a better visual reference of the path:

void OnDrawGizmos ()
{
    Vector3 direction;
    Color tmp = Gizmos.color;
    Gizmos.color = Color.magenta;//example color
    int i;
    for (i = 0; i < nodes.Count - 1; i++)
    {
        Vector3 src = nodes[i].transform.position;
        Vector3 dst = nodes[i+1].transform.position;
        direction = dst - src;
        Gizmos.DrawRay(src, direction);
    }
    Gizmos.color = tmp;
}

Avoiding agents

In crowd-simulation games, it would be unnatural to see agents behaving entirely like particles in a physics-based system. The goal of this recipe is to create an agent capable of mimicking our peer-evasion movement.

Getting ready

We need to create a tag called Agent and assign it to those game objects that we would like to avoid, and we also need to have the Agent script component attached to them.

Getting ready

An example of how should look the Inspector of a dummy agent to avoid

How to do it...

This recipe will require the creation and handling of just one file:

  1. Create the AvoidAgent behavior, which is composed of a collision avoidance radius and the list of agents to avoid:
    using UnityEngine;
    using System.Collections;
    using System.Collections.Generic;
    
    public class AvoidAgent : AgentBehaviour
    {
        public float collisionRadius = 0.4f;
        GameObject[] targets;
    }
  2. Implement the Start function in order to set the list of agents according to the tag we created earlier:
    void Start ()
    {
        targets = GameObject.FindGameObjectsWithTag("Agent");
    }
  3. Define the GetSteering function:
    public override Steering GetSteering()
    {
        // body
    }
  4. Add the following variables to compute distances and velocities from agents that are nearby:
    Steering steering = new Steering();
    float shortestTime = Mathf.Infinity;
    GameObject firstTarget = null;
    float firstMinSeparation = 0.0f;
    float firstDistance = 0.0f;
    Vector3 firstRelativePos = Vector3.zero;
    Vector3 firstRelativeVel = Vector3.zero;
  5. Find the closest agent that is prone to collision with the current one:
    foreach (GameObject t in targets)
    {
        Vector3 relativePos;
        Agent targetAgent = t.GetComponent<Agent>();
        relativePos = t.transform.position - transform.position;
        Vector3 relativeVel = targetAgent.velocity - agent.velocity;
        float relativeSpeed = relativeVel.magnitude;
        float timeToCollision = Vector3.Dot(relativePos, relativeVel);
        timeToCollision /= relativeSpeed * relativeSpeed * -1;
        float distance = relativePos.magnitude;
        float minSeparation = distance - relativeSpeed * timeToCollision;
        if (minSeparation > 2 * collisionRadius)
            continue;
        if (timeToCollision > 0.0f && timeToCollision < shortestTime)
        {
            shortestTime = timeToCollision;
            firstTarget = t;
            firstMinSeparation = minSeparation;
            firstRelativePos = relativePos;
            firstRelativeVel = relativeVel;
        }
    }
  6. If there is one, then get away:
    if (firstTarget == null)
        return steering;
    if (firstMinSeparation <= 0.0f || firstDistance < 2 * collisionRadius)
        firstRelativePos = firstTarget.transform.position;
    else
        firstRelativePos += firstRelativeVel * shortestTime;
    firstRelativePos.Normalize();
    steering.linear = -firstRelativePos * agent.maxAccel;
    return steering;

How it works...

Given a list of agents, we take into consideration which one is closest, and if it is close enough, we make it so the agent tries to escape from the expected route of that first one according to its current velocity so that they don't collide.

There's more

This behavior works well when combined with other behaviors using blending techniques (some are included in this chapter); otherwise it's a starting point for your own collision avoidance algorithms.

Avoiding walls

This technique aims at imitating our capacity to evade walls by considering a safety margin, and creating repulsion from their surfaces when that gap is broken.

Getting ready

This technique uses the RaycastHit structure and the Raycast function from the physics engine, so it's recommended that you take a refresher on the docs in case you're a little rusty on the subject.

How to do it...

Thanks to our previous hard work, this recipe is a short one:

  1. Create the AvoidWall behavior derived from Seek:
    using UnityEngine;
    using System.Collections;
    
    public class AvoidWall : Seek
    {
        // body
    }
  2. Include the member variables for defining the safety margin, and the length of the ray to cast:
    public float avoidDistance;
    public float lookAhead;
  3. Define the Awake function to set up the target:
    public override void Awake()
    {
        base.Awake();
        target = new GameObject();
    }
  4. Define the GetSteering function for the following steps:
    public override Steering GetSteering()
    {
        // body
    }
  5. Declare and set the variable needed for ray casting:
    Steering steering = new Steering();
    Vector3 position = transform.position;
    Vector3 rayVector = agent.velocity.normalized * lookAhead;
    Vector3 direction = rayVector;
    RaycastHit hit;
  6. Cast the ray and make the proper calculations if a wall is hit:
    if (Physics.Raycast(position, direction, out hit, lookAhead))
    {
        position = hit.point + hit.normal * avoidDistance;
        target.transform.position = position;
        steering = base.GetSteering();
    }
    return steering;

How it works...

We cast a ray in front of the agent; when the ray collides with a wall, the target object is placed in a new position taking into consideration its distance from the wall and the safety distance declared and delegating the steering calculations to the Seek behavior; this creates the illusion of the agent avoiding the wall.

There's more...

We could extend this behavior by adding more rays, like whiskers, in order to get better accuracy. Also, it is usually paired with other movement behaviors, such as Pursue, using blending.

There's more...

The original ray cast and possible extensions for more precise wall avoidance

See also

For further information on the RaycastHit structure and the Raycast function, please refer to the official documentation available online at:

Blending behaviors by weight

Blending techniques allow you to add behaviors and mix them without creating new scripts every time you need a new type of hybrid agent.

This is one of the most powerful techniques in this chapter, and it's probably the most used behaviour-blending approach because of its power and the low cost of implementation.

Getting ready

We must add a new member variable to our AgentBehaviour class called weight and preferably assign a default value—in this case, 1.0f. Besides this, we should refactor the Update function to incorporate weight as a parameter to the Agent class' SetSteering function. All in all, the new AgentBehaviour class should look something like this:

public class AgentBehaviour : MonoBehaviour
{
    public float weight = 1.0f;

    // ... the rest of the class

    public virtual void Update ()
    {
        agent.SetSteering(GetSteering(), weight);
   }
}

How to do it...

We just need to change the SetSteering agent function's signature and definition:

public void SetSteering (Steering steering, float weight)
{
    this.steering.linear += (weight * steering.linear);
    this.steering.angular += (weight * steering.angular);
}

How it works...

The weights are used to amplify the steering behavior result, and they're added to the main steering structure.

There's more...

The weights don't necessarily need to add up to 1.0f. The weight parameter is a reference for defining the relevance that the steering behavior will have among the other ones.

See also

In this project, there is an example of avoiding walls, worked out using weighted blending.

Blending behaviors by priority

Sometimes, weighted blending is not enough because heavyweight behaviors dilute the contributions of the lightweights, but those behaviors need to play their part too. That's when priority-based blending comes into play, applying a cascading effect from high-priority to low-priority behaviors.

Getting ready

The approach is very similar to the one used in the previous recipe. We must add a new member variable to our AgentBehaviour class. We should also refactor the Update function to incorporate priority as a parameter to the Agent class' SetSteering function. The new AgentBehaviour class should look something like this:

public class AgentBehaviour : MonoBehaviour
{
    public int priority = 1;
    // ... everything else stays the same
    public virtual void Update ()
    {
        agent.SetSteering(GetSteering(), priority);
    }
}

How to do it...

Now, we need to make some changes to the Agent class:

  1. Add a new namespace from the library:
    using System.Collections.Generic;
  2. Add the member variable for the minimum steering value to consider a group of behaviors:
    public float priorityThreshold = 0.2f;
  3. Add the member variable for holding the group of behavior results:
    private Dictionary<int, List<Steering>> groups;
  4. Initialize the variable in the Start function:
    groups = new Dictionary<int, List<Steering>>();
  5. Modify the LateUpdate function so that the steering variable is set by calling GetPrioritySteering:
    public virtual void LateUpdate ()
    {
        //  funnelled steering through priorities
        steering = GetPrioritySteering();
        groups.Clear();
        // ... the rest of the computations stay the same
        steering = new Steering();
    }
  6. Modify the SetSteering function's signature and definition to store the steering values in their corresponding priority groups:
    public void SetSteering (Steering steering, int priority)
    {
        if (!groups.ContainsKey(priority))
        {
            groups.Add(priority, new List<Steering>());
        }
        groups[priority].Add(steering);
    }
  7. Finally, implement the GetPrioritySteering function to funnel the steering group:
    private Steering GetPrioritySteering ()
    {
        Steering steering = new Steering();
        float sqrThreshold = priorityThreshold * priorityThreshold;
        foreach (List<Steering> group in groups.Values)
        {
            steering = new Steering();
            foreach (Steering singleSteering in group)
            {
                steering.linear += singleSteering.linear;
                steering.angular += singleSteering.angular;
            }
            if (steering.linear.sqrMagnitude > sqrThreshold ||
                    Mathf.Abs(steering.angular) > priorityThreshold)
            {
                return steering;
            }
    }

How it works...

By creating priority groups, we blend behaviors that are common to one another, and the first group in which the steering value exceeds the threshold is selected. Otherwise, steering from the least-priority group is chosen.

There's more...

We could extend this approach by mixing it with weighted blending; in this way, we would have a more robust architecture by getting extra precision on the way the behaviors make an impact on the agent in every priority level:

foreach (Steering singleSteering in group)
{
    steering.linear += singleSteering.linear * weight;
    steering.angular += singleSteering.angular * weight;
}

See also

There is an example of avoiding walls using priority-based blending in this project.

Combining behaviors using a steering pipeline

This is a different approach to creating and blending behaviors that is based on goals. It tries to be a middle-ground between movement-blending and planning, without the implementation costs of the latter.

Getting ready

Using a steering pipeline slightly changes the train of thought used so far. We need to think in terms of goals, and constraints. That said, the heavy lifting rests on the base classes and the derived classes that will define the behaviors; we need to start by implementing them.

The following code is for the Targeter class. It can be seen as a goal-driven behavior:

using UnityEngine;
using System.Collections;

public class Targeter : MonoBehaviour
{
    public virtual Goal GetGoal()
    {
        return new Goal();
    }
}

Now, we create the Decomposer class:

using UnityEngine;
using System.Collections;

public class Decomposer : MonoBehaviour
{
    public virtual Goal Decompose (Goal goal)
    {
        return goal;
    }
}

We also need a Constraint class:

using UnityEngine;
using System.Collections;

public class Constraint : MonoBehaviour
{
    public virtual bool WillViolate (Path path)
    {
        return true;
    }

    public virtual Goal Suggest (Path path) {
        return new Goal();
    }
}

And finally, an Actuator class:

using UnityEngine;
using System.Collections;

public class Actuator : MonoBehaviour
{
    public virtual Path GetPath (Goal goal)
    {
        return new Path();
    }

    public virtual Steering GetOutput (Path path, Goal goal)
    {
        return new Steering();
    }
}

How to do it...

The SteeringPipeline class makes use of the previously implemented classes in order to work, maintaining the component-driven pipeline but with a different approach, as mentioned earlier:

  1. Create the SteeringPipeline class deriving from the Wander behavior, including the array of components that it handles:
    using UnityEngine;
    using System.Collections;
    using System.Collections.Generic;
    
    public class SteeringPipeline : Wander
    {
        public int constraintSteps = 3;
        Targeter[] targeters;
        Decomposer[] decomposers;
        Constraint[] constraints;
        Actuator actuator;
    }
  2. Define the Start function to set the references to the attached components in the game object:
    void Start ()
    {
        targeters = GetComponents<Targeter>();
        decomposers = GetComponents<Decomposer>();
        constraints = GetComponents<Constraint>();
        actuator = GetComponent<Actuator>();
    }
  3. Define the GetSteering function to work out the goal and the steering value to reach it:
    public override Steering GetSteering()
    {
        Goal goal = new Goal();
        foreach (Targeter targeter in targeters)
            goal.UpdateChannels(targeter.GetGoal());
        foreach (Decomposer decomposer in decomposers)
            goal = decomposer.Decompose(goal);
        for (int i = 0; i < constraintSteps; i++)
        {
            Path path = actuator.GetPath(goal);
            foreach (Constraint constraint in constraints)
            {
                if (constraint.WillViolate(path))
                {
                    goal = constraint.Suggest(path);
                    break;
                }
                return actuator.GetOutput(path, goal);
            }
        }
        return base.GetSteering();
    }

How it works...

This code takes a composite goal generated by targeters, creates sub-goals using decomposers, and evaluates them to comply with defined constraints before "blending" them into a final goal in order to produce a steering result. If everything fails (the constraints are not satisfied), it uses the default Wander behavior.

There's more...

You should try to implement some of the behavior recipes in terms of targeters, decomposers, constraints, and an actuator. Take into account that there's room for one actuator only, and it's the one responsible for making the final decision. A good example is as follows:

  • Targeters: seeking, arriving, facing, and matching velocity
  • Decomposers: path-finding algorithms
  • Constraints: avoiding walls/agents

See also

For more theoretical insights, refer to Ian Millington's book, Artificial Intelligence for Games.

Shooting a projectile

This is the stepping stone for scenarios where we want to have control over gravity-reliant objects, such as balls and grenades, so we can then predict the projectile's landing spot, or be able to effectively shoot a projectile at a given target.

Getting ready

This recipe differs slightly as it doesn't rely on the base AgentBehaviour class.

How to do it...

  1. Create the Projectile class along with its member variables to handle the physics:
    using UnityEngine;
    using System.Collections;
    
    public class Projectile : MonoBehaviour
    {
        private bool set = false;
        private Vector3 firePos;
        private Vector3 direction;
        private float speed;
        private float timeElapsed;
    }
  2. Define the Update function:
    void Update ()
    {
        if (!set)
            return;
        timeElapsed += Time.deltaTime;
        transform.position = firePos + direction * speed * timeElapsed;
        transform.position += Physics.gravity * (timeElapsed * timeElapsed) / 2.0f;
        // extra validation for cleaning the scene
        if (transform.position.y < -1.0f)
            Destroy(this.gameObject);// or set = false; and hide it
    }
  3. Finally, implement the Set function in order to fire the game object (for example, calling it after it is instantiated in the scene):
    public void Set (Vector3 firePos, Vector3 direction, float speed)
    {
        this.firePos = firePos;
        this.direction = direction.normalized;
        this.speed = speed;
        transform.position = firePos;
        set = true;
    }

How it works...

This behavior uses high-school physics in order to generate the parabolic movement.

There's more...

We could also take another approach: implementing public properties in the script or declaring member variables as public and, instead of calling the Set function, having the script disabled by default in the prefab and enabling it after all the properties have been set. That way, we could easily apply the object pool pattern.

See also

For further information on the object pool pattern, please refer to the following Wikipedia article and an official Unity Technologies video tutorial available online at the following addresses:

Predicting a projectile's landing spot

After a projectile is shot, some agents need to make a run for it, if we're talking about a grenade, or look at it when we're developing a sports game. In either case, it's important to predict the projectile's landing spot in order to make decisions:

Getting ready

Before we get into predicting the landing position, it's important to know the time left before it hits the ground (or reaches a certain position). Thus, instead of creating new behaviors, we need to update the Projectile class.

How to do it...

  1. First, we need to add the GetLandingTime function to compute the landing time:
    public float GetLandingTime (float height = 0.0f)
    {
        Vector3 position = transform.position;
        float time = 0.0f;
        float valueInt = (direction.y * direction.y) * (speed * speed);
        valueInt = valueInt - (Physics.gravity.y * 2 * (position.y - height));
        valueInt = Mathf.Sqrt(valueInt);
        float valueAdd = (-direction.y) * speed;
        float valueSub = (-direction.y) * speed;
        valueAdd = (valueAdd + valueInt) / Physics.gravity.y;
        valueSub = (valueSub - valueInt) / Physics.gravity.y;
        if (float.IsNaN(valueAdd) && !float.IsNaN(valueSub))
            return valueSub;
        else if (!float.IsNaN(valueAdd) && float.IsNaN(valueSub))
            return valueAdd;
        else if (float.IsNaN(valueAdd) && float.IsNaN(valueSub))
            return -1.0f;
        time = Mathf.Max(valueAdd, valueSub);
        return time;
    }
  2. Now, we add the GetLandingPos function to predict the landing spot:
    public Vector3 GetLandingPos (float height = 0.0f)
    {
        Vector3 landingPos = Vector3.zero;
        float time = GetLandingTime();
        if (time < 0.0f)
            return landingPos;
        landingPos.y = height;
        landingPos.x = firePos.x + direction.x * speed * time;
        landingPos.z = firePos.z + direction.z * speed * time;
        return landingPos;
    }

How it works...

First, we solve the equation from the previous recipe for a fixed height and, given the projectile's current position and speed, we are able to get the time at which the projectile will reach the given height.

There's more...

Take into account the NaN validation. It's placed that way because there may be two, one, or no solution to the equation. Furthermore, when the landing time is less than zero, it means the projectile won't be able to reach the target height.

Targeting a projectile

Just like it's important to predict a projectile's landing point, it's also important to develop intelligent agents capable of aiming projectiles. It wouldn't be fun if our rugby-player agents aren't capable of passing the ball.

Getting ready

Just like the previous recipe, we only need to expand the Projectile class.

How to do it...

Thanks to our previous hard work, this recipe is a real piece of cake:

  1. Create the GetFireDirection function:
    public static Vector3 GetFireDirection (Vector3 startPos, Vector3 endPos, float speed)
    {
        // body
    }
  2. Solve the corresponding quadratic equation:
    Vector3 direction = Vector3.zero;
    Vector3 delta = endPos - startPos;
    float a = Vector3.Dot(Physics.gravity, Physics.gravity);
    float b = -4 * (Vector3.Dot(Physics.gravity, delta) + speed * speed);
    float c = 4 * Vector3.Dot(delta, delta);
    if (4 * a * c > b * b)
        return direction;
    float time0 = Mathf.Sqrt((-b + Mathf.Sqrt(b * b - 4 * a * c)) / (2*a));
    float time1 = Mathf.Sqrt((-b - Mathf.Sqrt(b * b - 4 * a * c)) / (2*a));
  3. If shooting the projectile is feasible given the parameters, return a non-zero direction vector:
    float time;
    if (time0 < 0.0f)
    {
        if (time1 < 0)
            return direction;
        time = time1;
    }
    else
    {
        if (time1 < 0)
            time = time0;
        else
            time = Mathf.Min(time0, time1);
    }
    direction = 2 * delta - Physics.gravity * (time * time);
    direction = direction / (2 * speed * time);
    return direction;

How it works...

Given a fixed speed, we solve the corresponding quadratic equation in order to obtain the desired direction (when at least one time value is available), which doesn't need to be normalized because we already normalized the vector while setting up the projectile.

There's more...

Take into account that we are returning a blank direction when time is negative; it means that the speed is not sufficient. One way to overcome this is to define a function that tests different speeds and then shoots the projectile.

Another relevant improvement is to add an extra parameter of the type bool for those cases when we have two valid times (which means two possible arcs), and we need to shoot over an obstacle such as a wall:

if (isWall)
    time = Mathf.Max(time0, time1);
else
    time = Mathf.Min(time0, time1);

Creating a jump system

Imagine that we're developing a cool action game where the player is capable of escaping using cliffs and rooftops. In that case, the enemies need to be able to chase the player and be smart enough to discern whether to take the jump and gauge how to do it.

Getting ready

We need to create a basic matching-velocity algorithm and the notion of jump pads and landing pads in order to emulate a velocity math so that we can reach them.

Also, the agents must have the tag Agent, the main object must have a Collider component marked as trigger. Depending on your game, the agent or the pads will need the Rigidbody component attached.

The following is the code for the VelocityMatch behavior:

using UnityEngine;
using System.Collections;

public class VelocityMatch : AgentBehaviour {
    
    public float timeToTarget = 0.1f;

    public override Steering GetSteering()
    {
        Steering steering = new Steering();
        steering.linear = target.GetComponent<Agent>().velocity - agent.velocity;
        steering.linear /= timeToTarget;
        if (steering.linear.magnitude > agent.maxAccel)
            steering.linear = steering.linear.normalized * agent.maxAccel;

        steering.angular = 0.0f;
        return steering;
    }
}

Also, it's important to create a data type called JumpPoint:

using UnityEngine;

public class JumpPoint
{
    public Vector3 jumpLocation;
    public Vector3 landingLocation;
    //The change in position from jump to landing
    public Vector3 deltaPosition;

    public JumpPoint ()
        : this (Vector3.zero, Vector3.zero)
    {
    }

    public JumpPoint(Vector3 a, Vector3 b)
    {
        this.jumpLocation = a;
        this.landingLocation = b;
        this.deltaPosition = this.landingLocation - this.jumpLocation;
    }
}

How to do it...

We will learn how to implement the Jump behavior:

  1. Create the Jump class deriving from VelocityMatch, with its member variables:
    using UnityEngine;
    using System.Collections.Generic;
    
    public class Jump : VelocityMatch
    {
        public JumpPoint jumpPoint;
        //Keeps track of whether the jump is achievable
        bool canAchieve = false;
        //Holds the maximum vertical jump velocity
        public float maxYVelocity;
        public Vector3 gravity = new Vector3(0, -9.8f, 0);
        private Projectile projectile;
        private List<AgentBehaviour> behaviours;
        
        // next steps
    }
  2. Implement the Isolate method. It disables all the agent behaviors, except for the Jump component:
    public void Isolate(bool state)
    {
        foreach (AgentBehaviour b in behaviours)
            b.enabled = !state;
        this.enabled = state;
    }
  3. Define the function for calling the jumping effect, using the projectile behavior we learned before:
    public void DoJump()
    {
        projectile.enabled = true;
        Vector3 direction;
        direction = Projectile.GetFireDirection(jumpPoint.jumpLocation, jumpPoint.landingLocation, agent.maxSpeed);
        projectile.Set(jumpPoint.jumpLocation, direction, agent.maxSpeed, false);
    }
  4. Implement the member function for setting up the behaviors' target for matching its velocity:
    protected void CalculateTarget()
    {
        target = new GameObject();
        target.AddComponent<Agent>();
    
        //Calculate the first jump time
        float sqrtTerm = Mathf.Sqrt(2f * gravity.y * jumpPoint.deltaPosition.y + maxYVelocity * agent.maxSpeed);
        float time = (maxYVelocity - sqrtTerm) / gravity.y;
    
        //Check if we can use it, otherwise try the other time
        if (!CheckJumpTime(time))
        {
            time = (maxYVelocity + sqrtTerm) / gravity.y;
        }
    }
  5. Implement the function for computing the time:
    //Private helper method for the CalculateTarget function
    private bool CheckJumpTime(float time)
    {
        //Calculate the planar speed
        float vx = jumpPoint.deltaPosition.x / time;
        float vz = jumpPoint.deltaPosition.z / time;
        float speedSq = vx * vx + vz * vz;
    
        //Check it to see if we have a valid solution
        if (speedSq < agent.maxSpeed * agent.maxSpeed)
        {
            target.GetComponent<Agent>().velocity = new Vector3(vx, 0f, vz);
            canAchieve = true;
            return true;
        }
        return false;
    }
  6. Override the Awake member function. The most important thing here is caching the references to other attached behaviors, so Isolate function makes sense:
    public override void Awake()
    {
        base.Awake();
        this.enabled = false;
        projectile = gameObject.AddComponent<Projectile>();
        behaviours = new List<AgentBehaviour>();
        AgentBehaviour[] abs;
        abs = gameObject.GetComponents<AgentBehaviour>();
        foreach (AgentBehaviour b in abs)
        {
            if (b == this)
                continue;
            behaviours.Add(b);
        }
    }
  7. Override the GetSteering member function:
    public override Steering GetSteering()
    {
        Steering steering = new Steering();
        
        // Check if we have a trajectory, and create one if not.
        if (jumpPoint != null && target == null)
        {
            CalculateTarget();
        }
        //Check if the trajectory is zero. If not, we have no acceleration.
        if (!canAchieve)
        {
            return steering;
        }
    
        //Check if we've hit the jump point
        if (Mathf.Approximately((transform.position - target.transform.position).magnitude, 0f) &&
            Mathf.Approximately((agent.velocity - target.GetComponent<Agent>().velocity).magnitude, 0f))
        {
            DoJump();
            return steering;
        }
        return base.GetSteering();
    }

How it works...

The algorithm takes into account the agent's velocity and calculates whether it can reach the landing pad or not. The behavior's target is the one responsible for executing the jump, and if it judges that the agent can, it tries to match the targets' vertical velocity while seeking the landing pad's position.

There is more

We will need a jump pad and a landing pad in order to have a complete jumping system. Both the jump and landing pads need the Collider component marked as trigger. Also, as stated before, they will probably need to have a Rigidbody component, too, as seen in the image below.

There is more

The pads we will need a MonoBehaviour script attached as explained below.

The following code is to be attached to the jump pad:

using UnityEngine;

public class JumpLocation : MonoBehaviour
{
    public LandingLocation landingLocation;

    public void OnTriggerEnter(Collider other)
    {
        if (!other.gameObject.CompareTag("Agent"))
            return;
        Agent agent = other.GetComponent<Agent>();
        Jump jump = other.GetComponent<Jump>();
        if (agent == null || jump == null)
            return;
        Vector3 originPos = transform.position;
        Vector3 targetPos = landingLocation.transform.position;
        jump.Isolate(true);
        jump.jumpPoint = new JumpPoint(originPos, targetPos);
        jump.DoJump();
    }
}

The following code is to be attached to the landing pad:

using UnityEngine;

public class LandingLocation : MonoBehaviour
{
    public void OnTriggerEnter(Collider other)
    {
        if (!other.gameObject.CompareTag("Agent"))
            return;
        Agent agent = other.GetComponent<Agent>();
        Jump jump = other.GetComponent<Jump>();
        if (agent == null || jump == null)
            return;
        jump.Isolate(false);
        jump.jumpPoint = null;
    }
}

See Also

The Shooting a projectile recipe

Left arrow icon Right arrow icon
Download code icon Download Code

Key benefits

  • *Empower your agent with decision making capabilities using advanced minimaxing and Negamaxing techniques
  • Discover how AI can be applied to a wide range of games to make them more interactive.
  • *Instigate vision and hearing abilities in your agent through collider based and graph based systems

Description

Unity 5 comes fully packaged with a toolbox of powerful features to help game and app developers create and implement powerful game AI. Leveraging these tools via Unity’s API or built-in features allows limitless possibilities when it comes to creating your game’s worlds and characters. This practical Cookbook covers both essential and niche techniques to help you be able to do that and more. This Cookbook is engineered as your one-stop reference to take your game AI programming to the next level. Get to grips with the essential building blocks of working with an agent, programming movement and navigation in a game environment, and improving your agent's decision making and coordination mechanisms - all through hands-on examples using easily customizable techniques. Discover how to emulate vision and hearing capabilities for your agent, for natural and humanlike AI behaviour, and improve them with the help of graphs. Empower your AI with decision-making functions through programming simple board games such as Tic-Tac-Toe and Checkers, and orchestrate agent coordination to get your AIs working together as one.

Who is this book for?

This book is intended for those who already have a basic knowledge of Unity and are eager to get more tools under their belt to solve AI and gameplay-related problems.

What you will learn

  • * Use techniques such as A*and A*mbush to empower your agents with path finding capabilities.
  • * Create a representation of the world and make agents navigate it
  • * Construct decision-making systems to make the agents take different actions
  • * Make different agents coordinate actions and create the illusion of technical behavior
  • * Simulate senses and apply them in an awareness system
  • * Design and implement AI in board games such as Tic-Tac-Toe and Checkers
  • * Implement efficient prediction mechanism in your agents with algorithms such as N-Gram predictor and naïve Bayes classifier
  • * Understand and analyze how the influence maps work.
Estimated delivery fee Deliver to Singapore

Standard delivery 10 - 13 business days

S$11.95

Premium delivery 5 - 8 business days

S$54.95
(Includes tracking information)

Product Details

Country selected
Publication date, Length, Edition, Language, ISBN-13
Last updated date : Feb 11, 2025
Publication date : Mar 31, 2016
Length: 278 pages
Edition : 1st
Language : English
ISBN-13 : 9781783553570
Vendor :
Unity Technologies
Tools :

What do you get with Print?

Product feature icon Instant access to your digital copy whilst your Print order is Shipped
Product feature icon Paperback book shipped to your preferred address
Product feature icon Redeem a companion digital copy on all Print orders
Product feature icon Access this title in our online reader with advanced features
Product feature icon DRM FREE - Read whenever, wherever and however you want
OR
Modal Close icon
Payment Processing...
tick Completed

Shipping Address

Billing Address

Shipping Methods
Estimated delivery fee Deliver to Singapore

Standard delivery 10 - 13 business days

S$11.95

Premium delivery 5 - 8 business days

S$54.95
(Includes tracking information)

Product Details

Last updated date : Feb 11, 2025
Publication date : Mar 31, 2016
Length: 278 pages
Edition : 1st
Language : English
ISBN-13 : 9781783553570
Vendor :
Unity Technologies
Tools :

Packt Subscriptions

See our plans and pricing
Modal Close icon
$19.99 billed monthly
Feature tick icon Unlimited access to Packt's library of 7,000+ practical books and videos
Feature tick icon Constantly refreshed with 50+ new titles a month
Feature tick icon Exclusive Early access to books as they're written
Feature tick icon Solve problems while you work with advanced search and reference features
Feature tick icon Offline reading on the mobile app
Feature tick icon Simple pricing, no contract
$199.99 billed annually
Feature tick icon Unlimited access to Packt's library of 7,000+ practical books and videos
Feature tick icon Constantly refreshed with 50+ new titles a month
Feature tick icon Exclusive Early access to books as they're written
Feature tick icon Solve problems while you work with advanced search and reference features
Feature tick icon Offline reading on the mobile app
Feature tick icon Choose a DRM-free eBook or Video every month to keep
Feature tick icon PLUS own as many other DRM-free eBooks or Videos as you like for just S$6 each
Feature tick icon Exclusive print discounts
$279.99 billed in 18 months
Feature tick icon Unlimited access to Packt's library of 7,000+ practical books and videos
Feature tick icon Constantly refreshed with 50+ new titles a month
Feature tick icon Exclusive Early access to books as they're written
Feature tick icon Solve problems while you work with advanced search and reference features
Feature tick icon Offline reading on the mobile app
Feature tick icon Choose a DRM-free eBook or Video every month to keep
Feature tick icon PLUS own as many other DRM-free eBooks or Videos as you like for just S$6 each
Feature tick icon Exclusive print discounts

Frequently bought together


Stars icon
Total S$ 208.97
Unity 5.x Shaders and Effects Cookbook
S$74.99
Unity 5.x Animation Cookbook
S$66.99
Unity 5.x Game AI Programming Cookbook
S$66.99
Total S$ 208.97 Stars icon

Table of Contents

9 Chapters
1. Behaviors – Intelligent Movement Chevron down icon Chevron up icon
2. Navigation Chevron down icon Chevron up icon
3. Decision Making Chevron down icon Chevron up icon
4. Coordination and Tactics Chevron down icon Chevron up icon
5. Agent Awareness Chevron down icon Chevron up icon
6. Board Games AI Chevron down icon Chevron up icon
7. Learning Techniques Chevron down icon Chevron up icon
8. Miscellaneous Chevron down icon Chevron up icon
Index Chevron down icon Chevron up icon

Customer reviews

Rating distribution
Full star icon Full star icon Half star icon Empty star icon Empty star icon 2.8
(4 Ratings)
5 star 25%
4 star 0%
3 star 25%
2 star 25%
1 star 25%
Hugo Apr 28, 2016
Full star icon Full star icon Full star icon Full star icon Full star icon 5
This is really a Programming AI Cookbook, it is full of algorithms teaching how to create a game with AI, like in the book cover there is over than 70 recipes, this will enable you to achieve almost any kind of behavior that you want to implement, you will learn techniques from intelligent movement, navigation, avoid agents and walls, jump systems, decision making, find the shortest path, the list is very big, and you have to take a little time to understand the concepts, is good to have your Unity open with the book code examples to see the results, and most important try to implement your own behaviors based on the book, this way you can create new AI behaviors, learning with these examples you can create more professional games. I think this book is for people with at least a short experience working in Unity and C#, most of the book is implementing algorithms in code, the visual part is on the example code, but if you are a complete beginner this book can be good if you work hard to learn the basics on the other books like Learning C# in Unity and Unity 5 by example, or in the Unity documentation, tutorials and forum. As a cookbook I you recommend this book for anyone who work with game development in general, this is well written, have a great author and structure, can be a great tool to consult and design your game, If you have an idea and doesn't know where to start, I think that you can easily find here one recipe for a game mechanic to start prototyping your game, or to create a game feature during the game development or to create a post production, this book is a mind blow, you will start to have more ideas or you know where to start the ideas that you thought were more difficult before.
Amazon Verified review Amazon
Silverleaf Jul 28, 2016
Full star icon Full star icon Full star icon Empty star icon Empty star icon 3
I am a professional game developer and got this book because it seemed like a good reference to quickly glance at some boiler plate type code to get it up and running quickly in Unity for prototyping without a lot of fuss. While this book does serve pretty well in that capacity I also have a few gripes with it:1. There are a good deal of bugs I've found in the code and I've had to fix them (which is easy for me), BUT this is supposed to be a cookbook with supposedly already tested and stable code. Recipes shouldn't have runtime bugs that need to be fixed by the user before they will work correctly at all. Some of the code doesn't seem like it was ever even test run at all before publishing. As an example, I found several bugs in the A* code as well as the SetNeighbours() code in the get8 branch where that code has a basic bug that makes it only get 5 neighbors instead of the 8 it should be (the j loop was missing a + 1 on the max count). Then in addition to that, the same code can crash when indexing into the array using the 'c' value because based on the non-clamped values, the 'c' variable can become < 0 and causes an unhandled runtime exception. This is just one example of the types of issues I've run in to.2. I did not like that in order to find relevant code to one recipe, you have to thumb forward or backwards to other recipes you don't really care about in order to find missing method dependencies in the recipe you DO care about. This is a cookbook, but it reads more like one long tutorial spread over many sub-articles instead. It would have been better if all that code was maybe all put into a code index that was easily searchable by some tag id in the book like: Code Listing 1, etc. Instead I actually had to get manually 'search' other recipes and read them in order to track down missing code. :/3. The web link provided in the book for the GPWiki PriorityQueue is a dead link. The page doesn't even come up. So I had to just implement my own priority queue and shoehorn that in to the recipe code. (which was fun actually), but still, the book should make sure links on a book this new are actually valid?
Amazon Verified review Amazon
Amazon Customer Apr 05, 2017
Full star icon Full star icon Empty star icon Empty star icon Empty star icon 2
I found the code more confusing than helpful and somewhat disorganized. I've been coding AI most of my professional life (15+ yrs) and I found this book difficult to use. It might be okay as a book for beginners who don't really want to understand AI but just get something working, but I don't think it's even properly organized for that purpose. I'd recommend one of the Game AI Wisdom books or perhaps AI Game Programming Wisdom.
Amazon Verified review Amazon
Sparo Vigil & Mick Macha Mar 10, 2018
Full star icon Empty star icon Empty star icon Empty star icon Empty star icon 1
Coming from a programming background in other fields and expecting a reasonable handbook to implementing AI in Unity, I have been disappointed in this book on multiple fronts. The author seems to have only a cursory understanding of C# programming (littering his code with unnecessary and obfuscating behavior like initializing structs to their default value), rarely discusses the actual meaning of his source code in detail, and goes so far as to reinvent the wheel on multiple occasions. As an example, he completely ignores Unity's built-in tools for managing state machines, in turn choosing to write a less capable one himself. All of chapter three is spent on that, which is roughly an eighth of the book, with no mention of Mecanim cover to cover.It is clear to me that he is simply unprepared to write a guide on this subject, let alone a cookbook. If you want to understand Unity, anything is better, inclusive of the on-site documentation. If you want to understand soft AI as applied to games, I might suggest Programming Game AI by Example, by Mat Buckland; it uses C++ instead of C#, but the descriptions are much clearer. If you need an actual cookbook, I would strongly suggest getting any other one, even if it requires a broader scope than Unity alone; it will still save you time.
Amazon Verified review Amazon
Get free access to Packt library with over 7500+ books and video courses for 7 days!
Start Free Trial

FAQs

What is the digital copy I get with my Print order? Chevron down icon Chevron up icon

When you buy any Print edition of our Books, you can redeem (for free) the eBook edition of the Print Book you’ve purchased. This gives you instant access to your book when you make an order via PDF, EPUB or our online Reader experience.

What is the delivery time and cost of print book? Chevron down icon Chevron up icon

Shipping Details

USA:

'

Economy: Delivery to most addresses in the US within 10-15 business days

Premium: Trackable Delivery to most addresses in the US within 3-8 business days

UK:

Economy: Delivery to most addresses in the U.K. within 7-9 business days.
Shipments are not trackable

Premium: Trackable delivery to most addresses in the U.K. within 3-4 business days!
Add one extra business day for deliveries to Northern Ireland and Scottish Highlands and islands

EU:

Premium: Trackable delivery to most EU destinations within 4-9 business days.

Australia:

Economy: Can deliver to P. O. Boxes and private residences.
Trackable service with delivery to addresses in Australia only.
Delivery time ranges from 7-9 business days for VIC and 8-10 business days for Interstate metro
Delivery time is up to 15 business days for remote areas of WA, NT & QLD.

Premium: Delivery to addresses in Australia only
Trackable delivery to most P. O. Boxes and private residences in Australia within 4-5 days based on the distance to a destination following dispatch.

India:

Premium: Delivery to most Indian addresses within 5-6 business days

Rest of the World:

Premium: Countries in the American continent: Trackable delivery to most countries within 4-7 business days

Asia:

Premium: Delivery to most Asian addresses within 5-9 business days

Disclaimer:
All orders received before 5 PM U.K time would start printing from the next business day. So the estimated delivery times start from the next day as well. Orders received after 5 PM U.K time (in our internal systems) on a business day or anytime on the weekend will begin printing the second to next business day. For example, an order placed at 11 AM today will begin printing tomorrow, whereas an order placed at 9 PM tonight will begin printing the day after tomorrow.


Unfortunately, due to several restrictions, we are unable to ship to the following countries:

  1. Afghanistan
  2. American Samoa
  3. Belarus
  4. Brunei Darussalam
  5. Central African Republic
  6. The Democratic Republic of Congo
  7. Eritrea
  8. Guinea-bissau
  9. Iran
  10. Lebanon
  11. Libiya Arab Jamahriya
  12. Somalia
  13. Sudan
  14. Russian Federation
  15. Syrian Arab Republic
  16. Ukraine
  17. Venezuela
What is custom duty/charge? Chevron down icon Chevron up icon

Customs duty are charges levied on goods when they cross international borders. It is a tax that is imposed on imported goods. These duties are charged by special authorities and bodies created by local governments and are meant to protect local industries, economies, and businesses.

Do I have to pay customs charges for the print book order? Chevron down icon Chevron up icon

The orders shipped to the countries that are listed under EU27 will not bear custom charges. They are paid by Packt as part of the order.

List of EU27 countries: www.gov.uk/eu-eea:

A custom duty or localized taxes may be applicable on the shipment and would be charged by the recipient country outside of the EU27 which should be paid by the customer and these duties are not included in the shipping charges been charged on the order.

How do I know my custom duty charges? Chevron down icon Chevron up icon

The amount of duty payable varies greatly depending on the imported goods, the country of origin and several other factors like the total invoice amount or dimensions like weight, and other such criteria applicable in your country.

For example:

  • If you live in Mexico, and the declared value of your ordered items is over $ 50, for you to receive a package, you will have to pay additional import tax of 19% which will be $ 9.50 to the courier service.
  • Whereas if you live in Turkey, and the declared value of your ordered items is over € 22, for you to receive a package, you will have to pay additional import tax of 18% which will be € 3.96 to the courier service.
How can I cancel my order? Chevron down icon Chevron up icon

Cancellation Policy for Published Printed Books:

You can cancel any order within 1 hour of placing the order. Simply contact customercare@packt.com with your order details or payment transaction id. If your order has already started the shipment process, we will do our best to stop it. However, if it is already on the way to you then when you receive it, you can contact us at customercare@packt.com using the returns and refund process.

Please understand that Packt Publishing cannot provide refunds or cancel any order except for the cases described in our Return Policy (i.e. Packt Publishing agrees to replace your printed book because it arrives damaged or material defect in book), Packt Publishing will not accept returns.

What is your returns and refunds policy? Chevron down icon Chevron up icon

Return Policy:

We want you to be happy with your purchase from Packtpub.com. We will not hassle you with returning print books to us. If the print book you receive from us is incorrect, damaged, doesn't work or is unacceptably late, please contact Customer Relations Team on customercare@packt.com with the order number and issue details as explained below:

  1. If you ordered (eBook, Video or Print Book) incorrectly or accidentally, please contact Customer Relations Team on customercare@packt.com within one hour of placing the order and we will replace/refund you the item cost.
  2. Sadly, if your eBook or Video file is faulty or a fault occurs during the eBook or Video being made available to you, i.e. during download then you should contact Customer Relations Team within 14 days of purchase on customercare@packt.com who will be able to resolve this issue for you.
  3. You will have a choice of replacement or refund of the problem items.(damaged, defective or incorrect)
  4. Once Customer Care Team confirms that you will be refunded, you should receive the refund within 10 to 12 working days.
  5. If you are only requesting a refund of one book from a multiple order, then we will refund you the appropriate single item.
  6. Where the items were shipped under a free shipping offer, there will be no shipping costs to refund.

On the off chance your printed book arrives damaged, with book material defect, contact our Customer Relation Team on customercare@packt.com within 14 days of receipt of the book with appropriate evidence of damage and we will work with you to secure a replacement copy, if necessary. Please note that each printed book you order from us is individually made by Packt's professional book-printing partner which is on a print-on-demand basis.

What tax is charged? Chevron down icon Chevron up icon

Currently, no tax is charged on the purchase of any print book (subject to change based on the laws and regulations). A localized VAT fee is charged only to our European and UK customers on eBooks, Video and subscriptions that they buy. GST is charged to Indian customers for eBooks and video purchases.

What payment methods can I use? Chevron down icon Chevron up icon

You can pay with the following card types:

  1. Visa Debit
  2. Visa Credit
  3. MasterCard
  4. PayPal
What is the delivery time and cost of print books? Chevron down icon Chevron up icon

Shipping Details

USA:

'

Economy: Delivery to most addresses in the US within 10-15 business days

Premium: Trackable Delivery to most addresses in the US within 3-8 business days

UK:

Economy: Delivery to most addresses in the U.K. within 7-9 business days.
Shipments are not trackable

Premium: Trackable delivery to most addresses in the U.K. within 3-4 business days!
Add one extra business day for deliveries to Northern Ireland and Scottish Highlands and islands

EU:

Premium: Trackable delivery to most EU destinations within 4-9 business days.

Australia:

Economy: Can deliver to P. O. Boxes and private residences.
Trackable service with delivery to addresses in Australia only.
Delivery time ranges from 7-9 business days for VIC and 8-10 business days for Interstate metro
Delivery time is up to 15 business days for remote areas of WA, NT & QLD.

Premium: Delivery to addresses in Australia only
Trackable delivery to most P. O. Boxes and private residences in Australia within 4-5 days based on the distance to a destination following dispatch.

India:

Premium: Delivery to most Indian addresses within 5-6 business days

Rest of the World:

Premium: Countries in the American continent: Trackable delivery to most countries within 4-7 business days

Asia:

Premium: Delivery to most Asian addresses within 5-9 business days

Disclaimer:
All orders received before 5 PM U.K time would start printing from the next business day. So the estimated delivery times start from the next day as well. Orders received after 5 PM U.K time (in our internal systems) on a business day or anytime on the weekend will begin printing the second to next business day. For example, an order placed at 11 AM today will begin printing tomorrow, whereas an order placed at 9 PM tonight will begin printing the day after tomorrow.


Unfortunately, due to several restrictions, we are unable to ship to the following countries:

  1. Afghanistan
  2. American Samoa
  3. Belarus
  4. Brunei Darussalam
  5. Central African Republic
  6. The Democratic Republic of Congo
  7. Eritrea
  8. Guinea-bissau
  9. Iran
  10. Lebanon
  11. Libiya Arab Jamahriya
  12. Somalia
  13. Sudan
  14. Russian Federation
  15. Syrian Arab Republic
  16. Ukraine
  17. Venezuela
Modal Close icon
Modal Close icon