A while ago, I wrote this article about a simple implementation of an Arcball camera in C++. In it, I described how that could get done without using Quaternions, spherical coordinates, or a lookAt function. Anyway, just out of curiosity, I decided to create a minimal implementation of an orbiting arcball camera in C# using spherical coordinates.
Conversion from spherical to cartesian coordinates and vice versa
Before I give you the code, let’s take a look at how you can convert cartesian coordinates to spherical coordinates and vice versa. By default, Unity uses cartesian coordinates to describe where objects are located in a virtual 2D or 3D space.
However, when you want to move an object on the surface of an invisible sphere, like in this case, it makes sense to convert the cartesian coordinates to spherical coordinates, which use the radius and two angles to define the position of the object on the sphere’s surface:

In figure 1, you can see how a point P gets mapped to the surface of a sphere. It can then get described using two angles (phi and theta) and the radius of the sphere. As a reference, you can also see the x, y, and z axes of the cartesian system.
Now, to transform a 3D point from the cartesian system to spherical coordinates, you can use the following formulas:

In C# code, the conversion can get implemented like this:
Vector3 getSphericalCoordinates(Vector3 cartesian)
{
float r = Mathf.Sqrt(
Mathf.Pow(cartesian.x, 2) +
Mathf.Pow(cartesian.y, 2) +
Mathf.Pow(cartesian.z, 2)
);
// use atan2 for built-in checks
float phi = Mathf.Atan2(cartesian.z / cartesian.x, cartesian.x);
float theta = Mathf.Acos(cartesian.y / r);
return new Vector3 (r, phi, theta);
}
As you can see, I used a vector to hold the three values that describe a point P in spherical coordinates. I, however, recommend that you write a custom class that holds the three values in suitably named fields.
Anyway, the conversion in the other direction is even simpler:

And, again, a C# helper method could look like this:
Vector3 getCartesianCoordinates(Vector3 spherical)
{
Vector3 ret = new Vector3 ();
ret.x = spherical.x * Mathf.Cos (spherical.z) * Mathf.Cos (spherical.y);
ret.y = spherical.x * Mathf.Sin (spherical.z);
ret.z = spherical.x * Mathf.Cos (spherical.z) * Mathf.Sin (spherical.y);
return ret;
}
Note that the axes’ names aren’t consistent between the formulas and the code because Unity uses a different reference compared to the mathematical formulas. Just keep this in mind when coding because it could save you some trouble and time!
A working barebones implementation
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
[RequireComponent (typeof (Camera))]
public class ArcballScript : MonoBehaviour
{
// Is true when the user wants to rotate the camera
bool ballEnabled = false;
float rotationSpeed = 1f;
float radius = 5f;
// The mouse cursor's position during the last frame
Vector3 last = new Vector3();
// The target that the camera looks at
Vector3 target = new Vector3 (0, 0, 0);
// The spherical coordinates
Vector3 sc = new Vector3 ();
void Start ()
{
this.transform.position = new Vector3 (radius, 0.0f, 0.0f);
this.transform.LookAt (target);
sc = getSphericalCoordinates (this.transform.position);
}
Vector3 getSphericalCoordinates(Vector3 cartesian)
{
float r = Mathf.Sqrt(
Mathf.Pow(cartesian.x, 2) +
Mathf.Pow(cartesian.y, 2) +
Mathf.Pow(cartesian.z, 2)
);
float phi = Mathf.Atan2(cartesian.z / cartesian.x, cartesian.x);
float theta = Mathf.Acos(cartesian.y / r);
if (cartesian.x < 0)
phi += Mathf.PI;
return new Vector3 (r, phi, theta);
}
Vector3 getCartesianCoordinates(Vector3 spherical)
{
Vector3 ret = new Vector3 ();
ret.x = spherical.x * Mathf.Cos (spherical.z) * Mathf.Cos (spherical.y);
ret.y = spherical.x * Mathf.Sin (spherical.z);
ret.z = spherical.x * Mathf.Cos (spherical.z) * Mathf.Sin (spherical.y);
return ret;
}
// Update is called once per frame
void Update ()
{
// Whenever the left mouse button is pressed, the
// mouse cursor's position is stored for the arc-
// ball camera as a reference.
if (Input.GetMouseButtonDown(0))
{
// last is a global vec3 variable
last = Input.mousePosition;
// This is another global variable
ballEnabled = true;
}
// When the user releases the left mouse button,
// all we have to do is to reset the flag.
if (Input.GetMouseButtonUp (0))
ballEnabled = false;
if (ballEnabled)
{
// Get the deltas that describe how much the mouse cursor got moved between frames
float dx = (last.x - Input.mousePosition.x) * rotationSpeed;
float dy = (last.y - Input.mousePosition.y) * rotationSpeed;
// Only update the camera's position if the mouse got moved in either direction
if (dx != 0f || dy != 0f)
{
// Rotate the camera left and right
sc.y += dx * Time.deltaTime;
// Rotate the camera up and down
// Prevent the camera from turning upside down (1.5f = approx. Pi / 2)
sc.z = Mathf.Clamp (sc.z + dy * Time.deltaTime, -1.5f, 1.5f);
// Calculate the cartesian coordinates for unity
transform.position = getCartesianCoordinates (sc) + target;
// Make the camera look at the target
transform.LookAt (target);
}
// Update the last mouse position
last = Input.mousePosition;
}
}
}
As you can see in the update method, it’s now easy to update the position of the camera if the mouse got moved. For that, you update the phi and theta angles in the spherical coordinate system and then convert it into a cartesian system, that unity can work with.
Now you just need to attach the script to your camera object, and you’re good to go!
Download the code
You can also view/download the code on GitHub!
Summary
Spherical coordinates are useful for describing a point on the surface of a sphere. In that system, you use the radius and two angles to specify a point. If you want to move the point on the sphere’s surface, you need to update the angles. Note that you have to work in radians and not degrees.
Image sources
[Fig. 1] Kugelkoordinaten, Ag2gaeh / CC BY-SA
[Fig. 2, Fig. 3] Screenshots from wikipedia.org

2 thoughts on “How to program an arcball (orbiting) camera in Unity using spherical coordinates”