# Polar Coordinates 🌀

## Cartesian Coordinate System

A coordinate system allows us to use numbers to determine the position of points in a 2D or 3D space. The most popular coordinate system is probably the Cartesian coordinate system. It allows you to locate each point by a pair of numerical coordinates `(x, y)`. There is a very good chance that you have used this system. It is everywhere — SVG, Canvas, WebGL and even Sketch & Illustrator.

## Polar Coordinate System

Polar coordinate system is a 2D coordinate system in which each point is determined by `r` & `θ`. Where `r` is the distance from the origin and `θ` is the angle from the x-axis.

### Converting Between Polar and Cartesian Coordinates

Trigonometry! Love it or hate, it is everywhere. You can convert a point from polar coordinates `(r, θ)` to Cartesian coordinates `(x, y)` using the equations below. These come in really handy since most tools only accept `x` & `y` as point locations.

``````const x = r * Math.cos(theta);
const y = r * Math.sin(theta);``````

## Patterns

Below I have a few animations by Dave Whyte aka bees & bombs. They all have one thing in common, they use polar coordinates to generate a pattern.

Cartesian coordinates are a great choice for placing things evenly on a rectangular grid. However, if you want to distribute things evenly around a circle then polar coordinates are generally the better option. Keeping the radius constant we compute the angle for each point as `angle = 360° * index / number of sides`.

Below is the JavaScript implementation of this idea. The `points` function also accepts an optional `offset` argument to shift the points along the perimeter of the circle. For example, the outermost circle of points in Figure 4 can be generated using `points(12, 200)`. The next one inwards using `points(12, 175, 15)`. The one after that using `points(12, 150, 30)` and so on. My splash CodePen is was built using this.

``````function points(count, radius, offset = 0) {
const angle = 360 / count;
const vertexIndices = range(count);

return vertexIndices.map(index => {
return {
theta: offset + degreesToRadians(offset + angle * index),
};
});
}

// number => [0, 1, 2, ... number]
function range(count) {
return Array.from(Array(count).keys());
}

return (Math.PI * angleInDegrees) / 180;
}``````

## Polygon Generator

A regular polygon is a polygon that is equiangular i.e., all its angles are equal and all its sides have the same length. This means that all the vertices of a regular polygon are points evenly spaced on a circle. And isn’t it handy that we just created a function that generates exactly this!

To generate an SVG polygon we will generate the list of of points using the `points` function and then simply connect the dots. With SVG we have two options for this: `<polygon>` or `<path>`. The example below generates the points attribute for a `<polygon>` element.

``````/**
* Usage with Vanilla JS/DOM:
*   polygonEl.setAttribute('points', polygon((5, 64, 18)));
*
* Usage with React:
*   <polygon points={polygon((5, 64, 18)} />
*
* Usage with View:
*   <polygon :points="polygon((5, 64, 18)" />
*/
.map(toCartesian)
.join(' ');
}

function toCartesian({ r, theta }) {
return [r * Math.cos(theta), r * Math.sin(theta)];
}``````

Seems simple but, you can do a lot of interesting things with it. Here are a couple of examples:

## Relative Polar Coordinates

When you define a point as `(r, θ)`, by default, this is relative to the origin `(0, 0)`. We can define points relative to other points by shifting the origin. This is often used for defining the position of curve handles relative to a vertex.

``````const x = cx + r * Math.cos(theta);
const y = cy + r * Math.sin(theta);``````

We can modify the polygon generator to allow us to draw a polygon centred at any location in the SVG canvas.

``````function polygon(noOfSides, circumradius, rotation, [cx = 0, cy = 0]) {
.map(pt => toCartesian(pt, [cx, cy]))
.join(' ');
}

function toCartesian({ r, theta }, [cx, cy]) {
return [cx + r * Math.cos(theta), cy + r * Math.sin(theta)];
}``````

## Rotation

Another common application of polar coordinates is to rotate things around a point. Here `(cx, cy)` is the point about which you want to rotate.

``````x = cx + r * Math.cos(theta);
y = cy + r * Math.sin(theta);

// somewhere in an animation loop
window.setInterval(() => {
theta++;
}, 1000 / 60);``````

## Polar curves

We started by looking at individual points. Then we grouped a few points into a set to define shapes. For this we used the polygon generator function to compute the location of each vertex of the shape. We can write similar functions using other mathematical equations. Allowing us to generate more complex shapes and curves.

Two dimensional curves are described by equations of the type `y = f(x)`. For example the equation of a circle is `x2 + y2 = r2`. We can generate the set of points, called locus, by iterating `x` and computing the corresponding `y` value or vice-versa. Therefore, each point will be of the form `(x, f(x))` or `(g(y), y)`.

With polar coordinates we can similarly draw polar curves. For example, the polar equation of a circle is `r = 2 * cos(0)`. The points on a polar curve have the form `(r(0), 0)`.

``````// examples of fn:
//   circle     : 2 * Math.cos(theta)
//   blob thing : a * (1 - Math.cos(theta) * Math.sin(3 * theta))
const r = fn(theta);

const x = cx + r * Math.cos(theta);
const y = cy + r * Math.sin(theta);``````

## Eukleides

All the diagrams in this post were created using a language called eukleides. It is a fantastic tool for making geometric drawings. Just look at this declarative API 😍

``````c = circle(point(3, 0), 3)
P = point(c, 170°)
M = point(0, 0)
N = point(6, 0)

draw (M.N.P)
label M, P, N right, 0.6``````

Questions, Comments or Suggestions? Open an Issue

### Creative coding from a front-end developer's perspective

My goal with this blog is to go beyond the basics. Breakdown my sketches. And make animation math approachable. The newsletter is more of that.

Hear about what's got my attention—new tools and techniques I've picked up. Experiments I'm working on. And previews of upcoming posts.