Simple Spaceship Controller

Home

Delta values

Before diving into the logic, it's important to know how delta values work. These are useful for controller because we have a universal language of how to translate movement using Vector values. Imagine a 2d topdown map and the 4 values below are your 'WASD' keys.

Now let's say we want to make this game playable on a gamepad, all we need to do is detect the direction of the equivelent up/down/left/right keys or even the thumbstick, and we can apply that vector to our player object's position. Say the player is at coordinates (3, 4), we add an Up value and now they're at (3, 5).

Updating position in Unity

Using the new input system, we read the input context as a vector2 and times (x) it by Time.SmoothDeltaTime. Since this is ran on every frame we hold the key down, the Delta time part makes sure the movement feels consistent regardless of the FPS you're getting. The code looks like this:

Vector2 deltaToMovement = context.ReadValuez<Vector2>() * Time.smoothDeltaTime;

The second step, now that we have a consistent movement speed, is to give it some strength. We don't want our spaceship moving at a snails pace!

deltaToMovement *= movementStrength;

In our unity file we should have defined a property to track the players input. For this example just define it as "private Vector2 playerInput = new Vector2();". Now, during each Update cycle, we want to check if theres any input - playerInput will be (0,0) if not - but if it is, then we need the following code:

if (playerInput != Vector2.zero) {     transform.position += playerInput; }

The Spaceship controller

For a spaceship we want to move around in a 3d space (no pun intended). On planets where gravity exists we don't allow characters to move up in a Y direction (the sky) or into the earth's crust. Since we're in space however, I've assigned up and down to the X rotation. Image the X-axis rotating similarly to a pig roast, except the pole is going through the left side of the spaceship through the right. This will cause the object nose to look up or down.

Ok we've figured out how to rotate up and down, but now we need to move the ship left and right.
You have 3 options here:
First option: Assign left and right buttons to Y rotation, this will simply rotate the nose in the expected direction.
Second option: Assign those turning values to the Z rotation axis and replicate your favourite star wars scenes
Third option: Both! However this will require 2 sets of movement keys (2 thumbsticks or WASD & arrow keys)

For this tutorial, we're going to go with the third option. In my first controller movement key set I've gone with turning the ship to face left or right using the Y rotation axis (Imagine the pole going straight down you body - head to toe). The forward and back keys in this case will make move the ship forwards or backwards. There's 2 separate methods for these inputs to clearly explain the code. OnForwardJets only listens to Up (W) and OnTurn listens for Left/Right (A/D)

public void OnForwardJets(InputAction.CallbackContext context) {     if (context.performed) jetsOn = true;     else if (context.canceled) jetsOn = false; } public void OnTurn(InputAction.CallbackContext context) {     if (context.action.inProgress)     {         Vector2 deltaValue = context.ReadValue() * Time.smoothDeltaTime;         deltaValue *= rotationStrength;         var correctValues = new Vector3(0, deltaValue.x, 0);         turnInput = correctValues;     } }

The above is my first set, bound to WASD, whereas this is the second set and will bind the the arrow keys. We have Up and Down for rotating the look up / look down motion, and the left right keys to spin on the Z axis so we can turn our ship on it's side to fit those wings through small gaps!

public void OnRotate(InputAction.CallbackContext context) {     if (context.action.inProgress)     {         Vector2 deltaValue = context.ReadValue<Vector2>() * Time.smoothDeltaTime;         deltaValue *= rotationStrength;         var correctValues = new Vector3(deltaValue.y, 0, -deltaValue.x);         rotateInput = correctValues;     } }

Conclusion

Finally, make sure your updating those values in your Update method like so

if (jetsOn) {     transform.position += (transform.forward * speed * Time.deltaTime); } if (rotateInput != Vector3.zero) {     transform.Rotate(rotateInput); } if (turnInput != Vector3.zero) {     transform.Rotate(turnInput); }

This controller gives complete freedom of moveability for the player, although at the start a little tricky to learn admittingly, but once you're use to moving in this fashion there will be no obstacles you can't get around!

Thank you for taking the time to read this article and good luck on your coding journey! Don't forget to check out my other Unity and coding tutorials.