Programming A Game With Unity A Beginner's Guide
Programming A Game With Unity A Beginner's Guide
Programming A
Game With
Unity: A
Beginner’s
Guide
Written by Andre Infante
This ebook is the intellectual property of MakeUseOf. It must only be published in its original
form. Using parts or republishing altered parts of this ebook is prohibited without permission
from MakeUseOf.com.
These indie game development teams have demonstrated an agility and risk-tolerance that, in
many cases, allows them to push gameplay innovation faster than their big budget
counterparts. A number of shockingly successful indie titles have premiered in recent years,
including Minecraft, Limbo, and Super Meat Boy, and even if you don’t have to skills to make
games like this, you can make a game using Buildbox.
In the rapidly evolving landscape of indie game development, Unity has emerged as something
of a de-facto standard: its low cost, ease of use, and broad feature set make it ideal for rapid
game development. Unity is so flexible, that you can even make your own custom game
controllers with a bit of DIY know how!
Even large studios such as CCP (Developers of Eve Online) use it for rapidly prototyping game
concepts. Unity provides a “game engine in a box” — a physics and rendering engine with hooks
for several scripting languages, adaptable to virtually any genre of video game.
While Unity does provide a visual editor for manipulating the game environment, Unity is not a
‘zero programming’ game creator tool. It requires you to program to produce results, but also
gives you a much more flexible and powerful tool than any ‘game maker’ program possibly
could.
Unity won’t do the work for you, but it does lower the barrier to entry substantially. Starting
completely from scratch with C++ and OpenGL, it can take days to get to the point where there’s
actually something rendered onscreen. Using Unity, it takes about ten seconds. Unity puts the
basic elements of game creation into the hands of novice programmers in a fast, intuitive way.
1. Versions of Unity
Unity comes in two basic flavors: the pro version and the free version. There are a number of
differences, but, broadly speaking, the pro version supports a number of visual improvements
(like real-time soft shadows and post-processing), and a large number of relatively minor
features that are extremely helpful for more complex games.
That said, for most relatively simple games you might want to build, the free version of Unity is
perfectly adequate. We’ll break down the key differences below in more detail for those
interested.
1.1 Pricing
The free version of Unity is, of course, free. However, there are a few limitations: the free version
of Unity cannot be licensed to any company with an annual income of more than $100,000.
While such organizations are beyond the scope of this guide, if you suspect you might become
such an organization, it’s probably wise to spring for the Pro version.
The Pro version of Unity is $75 a month, or $1500 for a permanent license, and has no limits on
what you can do with the games created with it. There is also a 30-day free trial available, which
we’ll be using for this guide, in order to give you as complete an overview of the available
features as possible. A one-year student license is also available through Studica for $129.
2. Installing Unity
Unity is straightforward to install. You can download the executable from unity3d.com/get-unity/
download.
Once downloaded, run it, and then follow the installer instructions. When the installation is
finished, a window titled ‘activate your Unity license’ will appear. Check the box marked
‘activate a free 30-day trial of Unity Pro’ and then ‘OK’.
Congratulations! You now have a 30-day trial of Unity Pro. When the trial expires, if you don’t
want to buy the pro version, you can switch to the free version and keep your existing content.
First off, if you’ve never programmed before, put this tutorial aside and spend a few days
working through Microsoft’s C# Language Primer until you feel comfortable using the language
for simple tasks.
If you’d like something a bit different to C# (but not necessarily a language you can use in
Unity), then take a look at our guide to the six easiest programming languages for beginners.
If you have programmed before in an imperative or object oriented language like C or Java, skim
the primer and familiarize yourself with how C# differs from other languages you’ve used in the
past. Either way, don’t proceed with the tutorial until you feel comfortable solving simple
problems with C# (for example, if I were to ask you to write a program that prints the first
hundred prime numbers, you should be able to write that program without consulting Google).
The most important concept to understand here is the object-oriented paradigm (abbreviated
as OOP). In object oriented languages, programs are divided into functional units
called Objects. Each object has its own private variables and functions. Object-specific
functions are called methods.
The idea here is modularity: by having each object isolated, and forcing other objects to interact
with it through its methods, you can reduce the number of possible unintentional interactions
— and, by extension, bugs. You can also create objects you can reuse later with no modification.
In Unity, you’ll be building these objects and attaching them to game entities (whose behavior
they’ll govern).
First, select the object and go to the Inspector. Click on the Add Component button.
Go to new script, enter the name you want, and click create and add.
using UnityEngine;
void Start () {
health = 100;
void Update(){
if (health > 0) {
• Using UnityEngine: This line tells C# that we want to use Unity’s libraries, which allow us
to connect to the Unity game engine.
• Public class Mook : MonoBehaviour:This line declares the class and its name — Mook.
• Private float health: This declares a private class variable (which can only be changed
from inside the class). The variable is given a value in Start.
• Void Start(): This declares a method called Start. Start is a special method that runs only
once, when the game initially launches.
• Void Update(): Update is another special method, which runs on every frame. Most of your
game logic will go here.
• //if you encounter the player on the road, kill him: This line is a comment (any line
starting with a double slash is ignored by C#). Comments are used to remind yourself of
what particular bits of code do. In this case, this comment is being used to stand in for a
more complicated block of code that actually does what the comment describes.
Along with Start and Update, you can instantiate your own methods with almost any name.
However, methods that you create won’t run unless they’re called. Let’s declare a method for a
hypothetical class called addTwoNumbers that adds two numbers together:
return a+b;
This declares a public (accessible to other objects) method that returns a float,
called addTwoNumbers, which takes two floats as input (called a and b). It then returns the
sum of the two values as its output.
Calling this method from within the same class (say, from inside Update) looks like this:
addTwoNumbers instance;
If your script is attached to an object that has special properties (like a particle emitter) that
can’t be accessed under the normal set of GameObject parameters, you can choose to treat it
as a different kind of game entity by using the GetComponent method.
GetComponent<ParticleSystem>().Play();
If any of this is unfamiliar to you, go back and go through the C# primer. It’ll save you a lot of
frustration as we proceed.
4. Unity Basics
In this section, we’re going to work our way through the basic mechanics of the Unity engine.
The workflow in Unity goes something like this:
1. Create an entity to serve a role in the game (blank GameObjects can be used for abstract
logical tasks).
2. Write or find a class file, and add it to the entity as a script (using the Add
Component button in the inspector view.
3. Run > test > debug > repeat until it works and move on to the next element of the game.
Unity comes with a number of basic view tabs that can be laid out in various ways to the taste
of the user. The big five are:
1. Game: displays a running instance of the game that you can interact with and test.
3. Inspector: allows you to modify individual entities in the game world by selecting them
in the editor tab.
4. Project: allows you to browse through the project’s files and drag models, materials, and
other resources into the editor tab to place them in the game world.
5. Hierarchy: this tab shows all objects in the world, allowing you to find distant objects in
the scene, and parent entities to one another by clicking and dragging.
See the diagram below for the locations of all these things:
4.1.1 Meshes
Meshes are the way 3D geometry is represented in Unity. You can either use Unity’s built-
in primitive objects (cubes, spheres, cylinders, etc), or import your own 3D models from a
modelling package like Blender or Maya. Unity supports a variety of 3D formats, including .fbx,
and .3ds.
The basic tools for manipulating meshes are the scaling, rotation, and translation buttons in
the upper left corner of the interface. These buttons add control icons to the models in the
editor view, which can then be used to manipulate them in space. To alter the texture or physics
properties of an object, select them and use the inspector view to analyze
the material and rigidbody elements.
In the hierarchy view, these gameplay elements can be dragged onto the main camera to make
them children, ensuring that they move and rotate with the camera.
GUI elements (text and textures) can have their size and scale adjusted using the relevant
fields in the inspector tab.
To import a texture, convert it to a .jpg, .png, or .bmp, and drag it into the assets folder under
the Unity project directory (which appears in My Documents by default). After a few seconds, a
loading bar will appear in the editor. When it finishes, you’ll be able to find the image as a
texture under the project tab.
4.1.4 Lights
Lights are GameObjects which project radiance onto the world. If there are no lights in your
scene, all polygons are drawn at the same brightness level, giving the world a flat, washed-out
look.
Lights can be positioned, rotated, and have several internal characteristics that you can
customize. The intensity slider controls the brightness of the light, and the range controls how
quickly it fades out.
The guidelines in the scene view show you the maximum range of the illumination. Play with
both settings to achieve the desired effect. You can also adjust the color of the light, the pattern
(cookie displayed on the surface the light is pointed at, and what kind of flare appears
onscreen when looking directly at the light. The cookie can be used to fake more realistic light
patterns, create dramatic false shadows, and simulate projectors.
The three main kinds of light are spot, point, and directional.
Spot lights have a location in 3D space and project light only in one direction in a cone of
variable angle. These are good for flashlights, searchlights, and, in general, give you more
precise control of lighting. Spot lights can cast shadows.
Point lights have a location in 3D space, and cast light evenly in all directions. Point lights do
not cast shadows.
Directional lights, finally, are used to simulate sunlight: they project light in a direction as
though from infinitely far away. Directional lights affect every object in the scene, and can
produce shadows.
There are a lot of parameters that you can tweak to achieve these effects, and you can access
them by spawning a particle system under the component editor > selecting the particle
system > opening the inspector tab. You can change the size, speed, direction, rotation, color,
and texture of each particle, and set most of those parameters to change over time as well.
Under the collision attribute, if you enable it and set the simulation space to world you’ll get
particles that will collide with objects in the world, which can be used for a number of realistic
particle effects, including rain, moving water, and sparks.
In this section, we’ll go over arranging the core elements — the scripting tutorial will come later.
First, let’s break down the game of Pong into its basic components. First, we need two paddles,
and a ball. The ball flies offscreen, so we’ll want a mechanism to reset it. We also want text to
display the current score, and, for the sake of showing you all the core elements of Unity, we’ll
want a fancy particle effect when you hit the ball. The whole game will need to be dramatically
lit.
That breaks down into a ball object (a sphere), a spawner, two paddle props with particle
emitters attached, a 3D-text entity, and a spot light. For this tutorial, we’ll be using the default
physic material bounce, with bounce combine set to multiply. Here’s what the setup looks like,
in ten screenshots:
Scale it appropriately, duplicate it, and put a sphere between the paddles for the ball.
Next, create two particle systems, pick the characteristics you want, and attach them to the
paddles.
Now that we’ve got that setup, we’re going to talk about what’s involved in scripting these
elements to make a game.
Keywords and comments are highlighted in blue and green, and numerical values and strings
appear in red. If you’ve used Eclipse or other IDE’s, MonoDevelop is very similar. You
can build your scripts from inside the editor, to check for syntax errors, like so:
In general, to get your script to interact with Unity, you’ll need to reference elements that the
object holding the script possesses (you can see a list of these elements under
the inspector tab when the relevant object is selected). You can then call methods or set
variables for each of these elements to enact the changes you want.
If you want a script on an object to affect the properties of a different object, you can create an
empty GameObject variable in your script, and use the inspector to assign it to another object
in the scene.
A list of the elements an object might have is as follows (taken from the inspector view of one
of our paddles in the above example):
1. Transform
3. Box Collider
4. Mesh Renderer
Each of these aspects of the object can be influenced from within a script. Next, we’ll look at
exactly how.
transform.position = newPositionVector3;
transform.rotation = newRotationQuaternion;
transform.localScale = newScaleVector3;
In the above examples, the named variables are of the types specified in the names. There are a
few key details here: position and scale are, as you’d expect, stored as Vector3s. You can access
the X, Y, and Z components of each (for example, transform.position.y gives you the distance
of an object above the zero plane).
However, to avoid gimbal lock, rotations are handled as Quaternions (four-component vectors).
Because hand-manipulating quaternions is unintuitive, you can manipulate rotations using
Eulerian angles by using the Quaternion.Euler method like so:
If you wish to move objects smoothly from one place to another, you’ll find the Slerp method for
quaternions and vector3s helpful. Slerp takes in three arguments – the current state, the final
state, and the speed of change, and smoothly interpolates between them at the given speed.
The syntax looks like this:
transform.position = Vector3.Slerp(startPositionVector3,
newDestinationVector3, 1);
6.2 Renderer
The renderer functions in Unity allow you to control the way the surfaces of props are rendered
on-screen. You can reassign the texture, change the color, and change the shader and visibility
of the object. The syntax looks like this:
renderer.enabled = false;
renderer.material.mainTexture = myTexture;
renderer.material.shader = newShader;
6.3 Physics
Unity comes with an integrated physics engine — something that physics sandbox games all use.
This allows you to assign the physical properties of objects and let the details of their
simulation be handled for you. In general, rather than trying to implement your own physics
using a textbook and the transform system, it is simpler and more robust to use Unity’s
physics engine to the greatest extent possible.
All physics props require colliders. However, the actual simulation itself is handled by
a rigidbody, which can be added in the inspector view. Rigidbodies can
be kinematic or nonkinematic.
Kinematic physics props collide with (and effect) nonkinematic physics props around them,
but are unaffected by collision themselves. Static kinematic props are the proverbial
immoveable objects, and moving kinematic objects are the proverbial unstoppable force (for
the record, when they collide, they simply pass through each other).
Beyond that, you can adjust the angular drag of the object (how much energy it takes to spin it),
change its mass, dictate whether or not it’s affected by gravity, and apply forces to it.
Examples:
rigidbody.angularDrag = 0.1f;
rigidbody.mass = 100;
rigidbody.isKinematic = false;
rigidbody.useGravity = true;
rigidbody.AddForce(transform.forward * 100);
These are all pretty self-explanatory. The only thing to note here is the use
of transform.forward. Vector3’s all have three components (.forward, .up, and .right)
associated with them, which can be accessed and rotates with them (forward is the direction
of the blue arrow in the editor). The transform.forward keyword is simply the forward vector for
the current object with magnitude 1. It can be multiplied by a float to create more force on the
object. You can also reference transform.up and transform.right, and negate them to get their
reverses.
There’s a certain amount of prep work needed to detect collisions in Unity. First, at least one of
the objects in the collision needs a non-kinematic rigidbody attached to it. Both objects must
have correct colliders, set to be non-triggers. The total speed of both objects must be low
enough that they actually collide, instead of simply skipping through one another.
If you’ve got all that taken care of, you can check for collision by placing a special collision
detection method in a script attached to the object you’d like to check collision with. The
method will look like this:
This method will automatically run during the first frame that another object touches your
object. The collision entity other is a reference to the object that you hit. You can, for example,
reference its gameobject, rigidbody, and transform characteristics to manipulate it in various
ways. While OnCollisionEnter is probably the most common function you’ll be using, you can
also use OnCollisionExit and OnCollisionStay (with otherwise identical syntax and usage),
which activate during the first frame that you stop colliding with an object and during every
frame that you’re colliding with an object, respectively.
Sometimes, it can also be useful to do what’s called raycasting. In raycasting, an infinitely thin
line (a ray) is cast through the world from some origin, along some vector, and, when it hits
something, the position and other details of the first collision are returned. The code for a
raycast looks like this:
RaycastHit hit;
This casts a ray from the position of the current object along -Vector3.up (straight down), and
links the variable hit to the first object it collides with. Once your ray has hit something, you
can access hit.distance to determine how far away it is, or hit.GameObject to manipulate the
object you hit.
Raycasts like this can be used for shooters to determine what the gun’s pointed at, or to select
objects when the camera looks at them, or for certain styles of movement mechanic.
The way you correct for this is by multiplying the values you’re using by the time it took to
render the last frame. This is done by using Time.deltaTime. This effectively changes the speed
of any variable you’re incrementing every frame from change per frame to change per second,
and you should probably make this change to any value you’re incrementing or decrementing
every frame.
2D sounds are appropriate for voice-overs and background music, and 3D sounds apply to
sounds generated by events in the world. In order to change whether or not a sound is 3D, select
it in the project view, switch to the inspector view and select the appropriate option from the
dropdown menu, then press the reimport button.
In order to actually play the sound, you’ll need to attach an audio source to a prop (the prop you
want the sound to originate from, in the case of a 3D sound). Then you’ll need to open the audio
clip field and select your sound file.
You can use myAudioSource.Pause() and myAudioSource.Play() to control those sound files.
You can adjust the falloff behaviors, volume, and doppler shifting of the sounds under
the inspector tab for the audio source.
6.7 Input
A game that doesn’t take any input from the user isn’t much of a game. There are a lot of
different kinds of input you can read in, and almost all of them are accessible through
the Input and KeyCodeobjects. Some sample input statements (which have a values evaluated
every frame) are below.
The functions of these lines is mostly self explanatory. Using these three kinds of input
reference, you can reconstruct the control schemes of most modern 3D computer games.
These bugs are typically not the most difficult to fix. What can be more problematic are subtle
semantic errors, in which you have successfully written a file full of valid C# – just not one that
does what you thought it would. If you have one of these errors, and you’re having trouble
tracking it down, there are a few things you can try to improve the situation.
The first is to pause the execution of the game, and check the console. You can pause the game
by clicking on the pause icon in the upper middle portion of the editor, and then
selecting console from the bottom of the window menu (or pressing Ctrl > Shift > C). Even if
there are no errors, warnings can still help to give some clues as to what might be going wrong.
If this doesn’t work, you can also try to get some idea about the state of your script by printing
the state of internal variables to validate that the program is doing what you think it’s doing.
You can use Debug.Log(String) to print the contents of a string to the console when the
program execution hits that line. In general, if you work backwards from what you think should
be happening through the things that should be making it happen, eventually you will reach a
point where your debug prints don’t do what you expect them to do. That’s where your error is.
Below, you can view the script for the ball with explanatory comments. The ball needs to
accomplish some basic goals: it should bounce in a complicated pattern, always maintaining
movement on both axes, and it should accelerate at a challenging but not impossible pace in
the horizontal direction.
BallHandler.cs
Next, we need to script our paddle, which you can view below. The paddle needs to move up and
down in response to key presses (but not outside certain bounds). It also needs to trigger the
particle system when it collides with something.
PaddleHandler.cs
Next, we need enemy AI: something that will cause the enemy’s paddle to track towards the ball
at a fixed rate. For that, we’ll be using Vector3.Slerp for maximum simplicity. We’d also like the
same particle behavior that we see on our own paddle.
EnemyAI.cs
Finally, we need a script to update the scoreboard and reset the ball when it goes out of bounds.
ScoreboardUpdater.cs
With those scripts attached and the references filled in, when we run our game of Pong, we
experience gameplay!
If you’re having trouble with the editor and interface of Unity, or just like video tutorials as a
matter of preference, there is a long list of high-quality Unity video tutorials available. More
extensive (but less broad) text tutorials for Unity are also available from CatLikeCoding.
Finally, if you have questions beyond the scope of documentation or tutorials, you can ask
specific questions at answers.Unity3d.com. Remember that answers are provided by volunteers,
so respect their time and search the database first to make sure your question hasn’t already
been answered.
For starters, know that you can only build an error-free project. To that end, make sure you have
the console open as you build: there are some error conditions that the game will ignore in the
editor, but will still abort an attempted build. This only dumps error messages to the console,
with no visible results onscreen, which can be frustrating if you forget to check. Once you’ve got
your game compiling error-free, though, you can select Build Settings under the File menu, or
press Ctrl > Shift
> B. This will bring up a simple dialog that allows you to build your game for several platforms.
Get the most crucial elements in first: have an idea in mind of your minimum viable product,
the simplest, most bare-bones thing you could possibly create and still feel as though you
achieved something worthwhile. Get to that minimum viable project before moving on to larger
ambitions.
This tutorial gives you a strong starting place, but the best way to learn Unity is by building a
game. Start building a game, fill gaps in your knowledge as they come up, and the gradual flow
of knowledge will erode away the things you don’t know surprisingly quickly.
Unity is a powerful tool, and with a bit of exploration, you can be building impressive
projects with it quicker than you might expect. Let us know what you built in the comment
section — we’d love to see!