Like so many things in CSS, the
transform property is surprisingly remarkable.
At first glance, it may seem like a pretty niche thing. How often do we need to rotate or skew something, after all? And yet, the more I learn about
transform, the more I find myself taking advantage of it. In my blog's codebase, I've used the
transform property more than 350 times!
In this blog post, we're diving deep into the
transform property. I'll show you some of the cool and unexpected things you can do with it!
Link to this headingTransform functions
transform property can do a whole bunch of different things, through the use of transform functions like
Let's look at each in turn.
Link to this headingTranslation
Translation allows us to move an item around:
We can use
translate to shift an item along in either axis:
x moves side to side,
y moves up and down. Positive values move down and to the right. Negative values move up and to the left.
Critically, the item's in-flow position doesn't change. As far as our layout algorithms are concerned, from Flow to Flexbox to Grid, this property has no effect.
For example: in this visualization, we have 3 children aligned using Flexbox. When we apply a transform to the middle child, the Flexbox algorithm doesn't notice, and keeps the other children in the same place:
This is similar to how
bottom work in positioned layout, with relatively-positioned elements.
When we want to move an element along a single axis, we can use
There's one thing that makes
translate ridiculously powerful, though. Something totally unique in the CSS language.
When we use a percentage value in
translate, that percentage refers to the element's own size, not the available space within the parent container.
transform: translateY(-100%) moves the box up by its exact height, no matter what that height is, to the pixel.
This is incredibly handy when we want an element to sit just outside another one:
A common usecase for this trick is to add a "close" button just outside a dialog box:
With the magic of
calc, we can even mix relative and absolute units:
This allows us to add a "buffer", so that we can translate something by its own size plus a few extra pixels.
Link to this headingScale
Alright, let's look at another transform function!
scale allows us to grow or shrink an element:
Scale uses a unitless value that represents a multiple, similar to
scale(2) means that the element should be 2x as big as it would normally be.
We can also pass multiple values, to scale the
y axis independently:
At first glance, this might seem equivalent to setting
height, but there's one big difference.
Check out what happens when our element has some text in it:
The text scales up and down with the element. We aren't just transforming the size and shape of the box, we're transforming the entire element and all of its descendants.
It may seem like a bummer that
scale will stretch/squash the element's contents, but we can actually use this effect to our advantage. For example, check out this old-timey TV power animation:
For this animation, the squashing effect actually improves the effect!
And, if we really don't want our text to squash, we can apply an inverse transform to the child.
This is an advanced technique, far beyond the scope of this blog post, but know that it's possible to use
scale to increase an element's size without distorting its children. Libraries like Framer Motion take advantage of this fact to build highly-performant animations without stretching or squashing.
Link to this headingRotate
You guessed it:
rotate will rotate our elements:
We typically use the
deg unit for rotation, short for degrees. But there's another handy unit we can use, one which might be easier to reason about:
turn unit represents how many turns the element should make. 1 turn is equal to 360 degrees.
It's obscure, but well-supported; the
turn unit goes all the way back to IE 9!
skew is a seldom-used but pretty-neat transformation:
translate, we can skew along either axis:
Skew can be useful for creating diagonal decorative elements (à la Stripe). With the help of
calc and some trigonometry, it can also be used on elements without distorting the text! This technique is explored in depth in Nils Binder's awesome blog post, “Create Diagonal Layouts Like It's 2020”.
Link to this headingTransform origin
Every element has an origin, the anchor that the transform functions execute from.
Check out how rotation changes when we tweak the transform origin:
The transform origin acts as a pivot point!
It isn't exclusive to rotation, either; here's how it affects scale:
This is useful for certain kinds of effects (for example, an element "growing out of" another one).
Link to this headingCombining multiple operations
We can string together multiple transform functions by space-separating them:
The order is important: the transform functions will be applied sequentially. Check out what happens if we reverse the order:
The transform functions are applied from right to left, like composition in functional programming.
In the first demo, we rotate the element in its natural position, and then translate it along the X axis.
In this second demo, however, we translate the element first. When we apply the rotation, it rotates around its origin, which hasn't changed.
Here's the same demo, but with the origin shown:
We can use this to our advantage:
In this example, we start by positioning the moon in the dead center of the planet. Our animation will shift it 80px to the right, and then cause it to rotate in a circle. Because the moon's origin is still in the center of the planet, it orbits around at a distance.
80px in the from/to blocks to see how it affects the animation!
Link to this headingInline elements
One common gotcha with transforms is that they don't work with inline elements in Flow layout.
Inline elements don't enjoy being jostled. Their goal is to wrap around some content with as little disruption as possible. Transforms aren't their cup of tea.
The easiest fix is to switch it to use
display: inline-block, or to use a different layout mode (eg. Flexbox or Grid).
Link to this headingThe third dimension
In addition to the 2D transforms we've covered in this tutorial, CSS can transform elements in a third dimension!
3D transforms have their own quirks and idiosyncracies. In order to do them justice, I'll be writing a separate post all about 3D transforms. Stay tuned!
Link to this headingGoing deeper
I have a confession to make: this tutorial wasn't originally written as a blog post. It's been ported from my upcoming CSS course:
The course is specifically created for folks who work with a JS framework like React or Angular or Vue. We cover the fundamentals of CSS, but within the context of the modern JS ecosystem.
It goes way deeper than my blog posts. There are over 150 videos, in addition to dozens of exercises and projects.
I've been working on it full-time for over a year now. Almost 5000 people purchased it in an early crowdfunding round, and their feedback has made the course so much better.
The course will be released on September 27th, and you can learn more here: css-for-js.dev.