Week Two
Week two was the beginning of refactoring the work of week one. If you remember I contained all of the functionality for the turtles movement and animation in one singular script. This was never the end goal, but with wanting to prototype a slight proof of concept - it did what it needed to even if it was not pretty.
I began this process by deciding what kind of state machine I wanted and decided that a hierarchy was the best option, over finite. This process involved basically splitting that script I wrote in week one into individual scripts allow me more control over state changes; much easier debugging and the main benefit WAY more control over the turtles behaviors within each individual state.
I also had to consider about the future of this Project when choosing the type, as I want to add swimming / diving into the game, that is much more suited to be withheld into a Hierarchal Machine - as there is no point having all the logic of walking and running within a swim state.
After considerable amounts of research via both books and tutorials. I managed to get enough to knowledge to attempt the state machine - Abstract State ( Base State Class holding all the functions that all the states will need in order to work - Enter State Exit State Update State etc etc ) Concrete states - all my states that have individual logic, but derive from the Abstract State. I will have a script set up handling all of the context within this state machine - this includes things like player inputs handling rotation etc. The concrete states will be fed the information by that script in order to interact with the logic within them, this includes when to swap states.
I began by creating one base class script ( Abstract State ) that all my states would inherit from. Split into different functions that each in there own allow me more control between the transitions and will make the debugging so much easier.
Then a few different (Concreate) states. Idle / Walk / Run / Jump / Grounded. Grounded will have a substate of Jump, whereas Idle has a Sub State of Walk which has a substate of Run. I then created a Script that handles all the context data that the concrete states use. Then a script which holds all the constructor functions which what needs to be run to create new instances of a class - the Base State Class.
Making sure the states switched between each other, was just copying the logic from the original script.
Lets Use the Jump State as a example, The State itself I do not want to be able to switch into a different state at all till the turtle has landed. Using the Context Data I mentioned earlier (ctx) and the Character Controllers built in IsGrounded? The script is not going to check incase it needs to switch until it is grounded.
The State Machine will then trigger the Exit State function within the Jump State script and in turn set the Context IsJumping Boolean within the Context Script to false as well as the animation controller within that context script, meaning we now know the turtle has landed.
It took three days of solid work, plenty of tutorials and even more research to get this state machine working. Now I have a fully functional state machine that will allow me to do so many things with ease.
Add Extra States
Adding Audio
Debugging
Adding Overall Functionality
Adding Specific Functionality
Adding Extra States is now a breeze, all I need to do is add it to the Factory - that creates the new instances of the Concrete States, create another Concreate State based of the Base State by copy and pasting a existing one and changing the names - and then filling in each functions functionality.
Adding Audio into this is also a breeze - If I want a noise when the turtle hits the ground, I would put it in the ExitState function of the JumpState - if I want to player a sound when the turtle is walking, Id start that audio in the EnterState function and stop it in the ExitState function.
During my research I was also introduced into a new concept called Getters / Setters which is a much cleaner was of accessing variables in other classes, so I used this to transfer the context data into the concrete states.
This is now a system that I can comfortably add to for the remainder of this project - and should not have to change it. It is also a system I can use in any other project I need where a state machine could be needed. Functionality can be seen below.
Although nothing progress wise can be seen here, there is a lot more of a stable behind the scenes. While testing to see how the turtle handled slight inclines, I realized I wasn’t happy with how the turtle clipped into the ledge. Footage can be seen below.
I then decided to add another rather common aspect into the project that is seen in many other 3D Platformers - Surface Alignment. Now I originally thought this would not be much of a endeavor, but I was proven wrong - just as the HandleRotation function in week one took a while to get correct so did this. Although I can say it was a LOT more frustrating, as most of the time it broke the rotating of the turtle completely, or made the turtle flip upside down, or make the turtle do a handstand?
I knew I needed to include the surface alignment either within its own Function in the Abstract State, or within the Rotate Function. I originally thought I would need to tweak how I find the targetRotation within that Function which is a variable that is local to that function, so I decided to just include it within that to save me having to make 3 more member variables.
I fired a Raytrace down from the players position (transform.position) fired it downwards ( - transform.up ) it output the fired result called hit, I then set the Vector3 variable called newUp to hold the normal result of this Raytrace.
I originally thought I would only need to take the y value of this and set it to the positionToLookAt.y value, as the rotation works, all I need is the elevation which I just set to 0 as I wanted the turtle to stay flat, that did not work - I spent about 3 hours reading through each Quaternion. function within the Unity Documentation to try to get this to work, and I finally got there.
Is seems silly how long it took me to figure this out, but I have learnt a lot along the way. I was in the right direction originally, it was to do with the target rotation, I needed to set the Vector3.up value in Qauternion Target Rotation to look, at the position but also its Up Value.
Alongside this I added a line that before the rotation of the character was set when it was moving, that set the tilt angle of the character, getting the surface align to work. Ideally I would want this all in one, but I am not sure if that would even be possible and I have other more important features to work on this early in the project. It works, if I have time later on I will potentially come back and look at it again.