I am going to use the first person controller found in the standard asset, plus additional assets found here and a nice skybox downloadable here. As for our foe, I found this asset which suits our needs perfectly.
Setting up the scene
Start by creating the scene. I dropped the first person character controller on a plane, placed a skybox to make the level look pretty and finally position the axe as a child of the first person camera in the controller. This is what the game window looks like:And below is a srceenshot of the hierarchy as it looks after placing these objects in the scene:
You can see that the axe model is placed as a child of the FirstPersonCharacter, which is where the camera is.
Place a capsule collider on it and set it to trigger. You don't need to be super accurate, something like this should do it:
Since we are here, let's drag the goblin model in the scene too. That particular model looks a bit too big, so I scaled it down to 0.45 for each axis in its transform. Once in the scene, you can remove the script attached to it and place a capsule collider on it and make it fit nicely around the model. Also, the rigidbody should be checked as isKinematic.
This is pretty much all we need in the scene now. The goblin model already comes with some animation, which we are going to script later.
We should now have the player, with the axe, and our goblin, and they both should collide with each other. However, this is our first problem:
When we get close to the goblin model, the axe passes through it. Unless this is something you want to have, Also, if we had to use the collider on our weapon and not set it to trigger, it could be a little problematic for the player to move around, as it might get stuck in other models when rotating and so on.
There are a couple of ways we can fix this: we could write a shader for the axe in order to have it rendered on top of everything, or we can use 2 cameras. We are going with the latter.
Create a new camera and attach it to the FirstPersonCharacter and set the transform to 0,0,0. Let's call it OverlayCamera. The camera should now be in the exact position as the first person camera that comes with the FPScontroller.Now, in the inspector for this camera, set the Clear Flags to Depth Only, the Culling Mask to Axe, and the Depth to 1.
Now, the other camera, which is a component of FirstPersonCharacter. Leave the Clear flags to Skybox, change the Culling mask by removing the Axe layer, and leave the Depth to 0.There are a couple of ways we can fix this: we could write a shader for the axe in order to have it rendered on top of everything, or we can use 2 cameras. We are going with the latter.
Create a new camera and attach it to the FirstPersonCharacter and set the transform to 0,0,0. Let's call it OverlayCamera. The camera should now be in the exact position as the first person camera that comes with the FPScontroller.Now, in the inspector for this camera, set the Clear Flags to Depth Only, the Culling Mask to Axe, and the Depth to 1.
At this point, the OverlayCamera is rendering only the axe (or anything on the Axe layer) and it always draws on top of the FirstPersonCharacter camera, because of the Depth parameter of value 1.
No matter how close we get to the goblin, the axe will always be drawn on top.
Axe animation
We should now create an attack animation for the axe. Honestly, I'm really not the best person to show you how to properly create an animation in Unity, I'm still trying to master the art myself. However, I will show you a GIF picture of the final result so you can have an idea of what I am trying to achieve:This is the attack animation. I also created a couple of idle animation, you will see them in the playable scene. Later on, we will need to add animation events.
The animation controller for the axe is shown below:
There is a sub state machine which I use to trigger some random idle animations, not something we should be concerned about for our purpose. We should in fact focus on the Axe_Attack state, which holds the animation just created. This state can be triggered by Any State via the trigger Attack and will return to the Axe_Idle state once the animation has reached the end. And don't forget to remove Loop from the Axe_Attack animation.
Here are the inspector screenshots of the 2 transition states side by side:
We can now start writing some scripts.
Scripting time
Create a C# script called Axe.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 | public class Axe : MonoBehaviour { Animator anim; CapsuleCollider cc; // Use this for initialization void Start () { anim = GetComponent<Animator> (); cc = GetComponent<CapsuleCollider> (); cc.enabled = false; } // Update is called once per frame void Update () { if (Input.GetMouseButtonDown (0)) anim.SetTrigger ("Attack"); } public void ActivateCollider(int active) { if (active == 0) cc.enabled = false; else cc.enabled = true; } void OnTriggerEnter(Collider c) { if (c.gameObject.GetComponent<IAxeHittable>()!=null) { cc.enabled = false;
c.gameObject.SendMessage ("OnGetHitByAxe", 5);
}
}
|
Pretty simple, after getting all our references we check in the Update function for any mouse click to trigger the attack animation.
The ActivateCollider method is called by animation events: when the axe is about half way extended, we activate the collider in order to trigger the collision. As the axe is animating back to idle position, we deactivate it.
This is necessary, otherwise every time we get close to the goblin it will register a collision, even if we are not attacking.
Also, we deactivate the collider immediately after hitting.
To detect whether the weapon has hit something we use interfaces. If the object collided with implements the interface IAxeHittable, we send a message to it, calling the method OnGetHitByAxe.
This is the interface:
1 2 3 4 | public interface IAxeHittable { void OnGetHitByAxe(float hitValue); } |
Now, we can create a Goblin script which will implement the interface and attach it to our goblin model.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | public class Goblin : MonoBehaviour, IAxeHittable { // Use this for initialization void Start () { } // Update is called once per frame void Update () { } public void OnGetHitByAxe(float hitValue) { Debug.Log ("Goblin got hit by Axe!"); } } |
As you can see, we MUST implement the method OnGetHitByAxe, which we'll use for handling damage to the enemy. At this point, all we do is to print to the console.
Conclusion
This is the basic structure on how to create some simple melee combat mechanics. As always, this can be improved to become a much more efficient and flexible system, which can be as complex or simple as your project would require.Example scene here