title | layout | slug | redirect_from | ||||
---|---|---|---|---|---|---|---|
Cube |
doc |
cube |
|
Cards are a good start for working with 3D transforms, but they only show off 3D in transition. To show off 3D at rest, we'll have to create true 3D objects: prisms. We'll start with a cube.
The markup for the cube is similar to the card. This time we need 6 child elements for all 6 faces of the cube.
{% highlight html %}
Basic position and size styles set the 6 faces on top of one another in the container.
{% highlight css %} .scene { width: 200px; height: 200px; perspective: 600px; }
.cube { width: 100%; height: 100%; position: relative; transform-style: preserve-3d; }
.cube__face { position: absolute; width: 200px; height: 200px; } {% endhighlight %}
Now all the faces are placed on top of one another, ready to be rotated. .cube__face--top
and .cube__face--bottom
will use rotateX()
so they are rotated around the vertical X axis.
{% highlight css %} .cube__face--front { transform: rotateY( 0deg); } .cube__face--right { transform: rotateY( 90deg); } .cube__face--back { transform: rotateY(180deg); } .cube__face--left { transform: rotateY(-90deg); } .cube__face--top { transform: rotateX( 90deg); } .cube__face--bottom { transform: rotateX(-90deg); } {% endhighlight %}
(We could remove the rotateY( 0deg)
style, as this transform has no effect, but let's leave it in for consistency.)
Now that faces are rotated, only the front and back faces are visible. The 4 side faces are all perpendicular to the viewer, so they appear on-edge, near-invisible. To push them out to their appropriate sides, they need to be translated out from the center of their positions. Each side of the cube is 200px wide. From the cube's center they'll need to be translated out half that distance, 100px
.
{% highlight css %} .cube__face--front { transform: rotateY( 0deg) translateZ(100px); } .cube__face--right { transform: rotateY( 90deg) translateZ(100px); } .cube__face--back { transform: rotateY(180deg) translateZ(100px); } .cube__face--left { transform: rotateY(-90deg) translateZ(100px); } .cube__face--top { transform: rotateX( 90deg) translateZ(100px); } .cube__face--bottom { transform: rotateX(-90deg) translateZ(100px); } {% endhighlight %}
Note here that the translate
function comes after the rotate
. The order of transform functions is meaningful. Each face is first rotated towards its position, then translated outward in a separate direction.
We have rendered a cube, but we're not done yet.
Take another look at the text "front" in the above cube. It's fuzzy.
3D transforms affect text rendering. When you apply a 3D transform, browsers take a snap-shot of the element and then re-render those pixels with 3D transforms applied. As such, fonts don't have the same anti-aliasing given their transformed size.
font-size: 2.5em
transform: scale(2.5)
transform: perspective(500px) translateZ(250px)
For the sake of our users, 3D transforms should not distort the interface. To resolve the distortion and restore pixel perfection to our cube, we can push back the 3D object, so that the front face will be positioned back at the Z origin.
{% highlight css %} .cube { transform: translateZ(-100px); } {% endhighlight %}
To expose any face of the cube, we'll need a style that rotates the cube to show that face. The transform is the opposite of the corresponding face. We toggle the necessary class on the .cube
to apply the appropriate transform.
{% highlight css %} .cube.show-front { transform: translateZ(-100px) rotateY( 0deg); } .cube.show-right { transform: translateZ(-100px) rotateY( -90deg); } .cube.show-back { transform: translateZ(-100px) rotateY(-180deg); } .cube.show-left { transform: translateZ(-100px) rotateY( 90deg); } .cube.show-top { transform: translateZ(-100px) rotateX( -90deg); } .cube.show-bottom { transform: translateZ(-100px) rotateX( 90deg); } {% endhighlight %}
Notice how the order of the transform functions has reversed. First we push the object back with translateZ
, then we rotate it.
Finishing up, we can add a transition to animate the rotation between states.
{% highlight css %} .cube { transition: transform 1s; } {% endhighlight %}
front right back left top bottom
function changeSide() { var checkedRadio = demo.querySelector(':checked'); var showClass = 'show-' + checkedRadio.value; if ( currentClass ) { cube.classList.remove( currentClass ); } cube.classList.add( showClass ); currentClass = showClass; } // set initial side changeSide();
demo.addEventListener( 'change', changeSide ); })(); </script>
{% include edit-codepen.html pen_slug="KRWjzm" %}