Mobile Web Wearable Web

CSS Transforms: Manipulating Elements

This tutorial demonstrates how you can use transforms with animations.

Warm-up

Become familiar with the CSS Transforms API basics by learning about:

Follow-up

Once we have learned the basics of CSS Transforms API, we can now move on to more advanced tasks, including:

Creating an Animation with Transforms

To enhance the user experience of your application, you must learn to use transforms with animations. This example uses the animation from the Creating a Logo Animation use case as a basis, and adds a more diverse visual effect to it with the transform property. In the modified animation:

  • As in the original animation, no elements are initially shown on the screen.
  • The Tizen logo gradually appears in the middle of the screen, and as it moves to the right, it becomes smaller. With the transform property, the logo is made to rotate.
  • Each letter in the word "TIZEN" consecutively comes in from the left of the screen and moves to the right to its correct location. With the transform property, the letters are translated in the 3D space and rotated around the Y axis.

Figure: Tizen logo to be transformed

Tizen logo to be transformed

  1. Create the HTML layout to control the movement of each individual animation element:
    <div class="animation-holder">
       <span class="tizen-txt t"></span>
       <span class="tizen-txt i"></span>
       <span class="tizen-txt z"></span>
       <span class="tizen-txt e"></span>
       <span class="tizen-txt n"></span>
       <span class="tizen-txt tm"></span>
       <span class="tizen-logo"></span>
    </div>
    
  2. Define the basic style of the animation elements. Add the perspective property to increase the Z axis effect of the 3D transform.
    .animation-holder 
    {
       -webkit-perspective: 1000px;
       height: 88px;
       left: 50%;
       margin: -54px 0px 0px -140px;
       position: absolute;
       top: 50%;
       width: 280px;
    }
    .tizen-txt,
    .tizen-logo 
    {
       background-position: 50% 50%;
       background-repeat: no-repeat;
       display: block;
       position: absolute;
    }
    .tizen-txt.t 
    {
       background-image: url("images/txt_t.png");
       height: 56px;
       left: 0px;
       top: 31px;
       width: 48px;
    }
    
  3. Create the animation:
    1. Assign keyframes for the logo element to transform it. In order to rotate the logo, use the -webkit-transform: rotate() function, which defines the angle of the rotation.

      @-webkit-keyframes tizen-logo 
      {
         0% 
         {
            -webkit-animation-timing-function: ease-in;
            height: 211px;
            left: 30px;
            opacity: 0;
            top: -61px;
            -webkit-transform: rotate(0deg);
            width: 220px;
         }
         30% 
         {
            -webkit-animation-timing-function: ease-out;
            height: 211px;
            left: 30px;
            opacity: 1;
            top: -61px;
            -webkit-transform: rotate(720deg);
            width: 220px;
         }
         50% 
         {
            height: 32px;
            left: 247px;
            opacity: 1;
            top: 0;
            -webkit-transform: rotate(1440deg);
            width: 33px;
         }
         100% 
         {
            height: 32px;
            left: 247px;
            opacity: 1;
            top: 0;
            -webkit-transform: rotate(1440deg);
            width: 33px;
         }
      }
      
      Note
      For a rotation, the image has to be carefully created to ensure the correct end result. The rotation occurs with the center of the element as the center. If the rotation center must be moved because the image center is not aligned, use the transform-origin property to adjust the rotation location.
    2. Create the keyframes for the first letter in the word "TIZEN". In the animation, due to the translate3d() and rotateY() methods, each letter transforms slightly from the right to the left as it comes in.

      Animation

      @-webkit-keyframes tizen-txt-t 
      {
         0% 
         {
            opacity: 0;
            -webkit-transform: translate3d(20px, 0, -200px) rotateY(90deg);
         }
         30% 
         {
            opacity: 0;
            -webkit-transform: translate3d(20px, 0, -200px) rotateY(90deg);
         }
         35% 
         {
            opacity: 1;
            -webkit-transform: translate3d(0, 0, 0) rotateY(0deg);
         }
         100% {}
      }
      
    3. To emphasize the fact that the letters are being created on the right, change the location of the transform. If the transform-origin property is declared for the entire animation element, the logo rotation changes. Consequently, you must only declared it for the letters.
      .tizen-txt 
      {
         -webkit-transform-origin: 100% 50%;
      }
      
    4. Create the keyframes for the other letters similarly:

      @-webkit-keyframes tizen-txt-i 
      {
         0% 
         {
            opacity: 0;
            -webkit-transform: translate3d(20px, 0, -200px) rotateY(90deg);
         }
         32% 
         {
            opacity: 0;
            -webkit-transform: translate3d(20px, 0, -200px) rotateY(90deg);
         }
         37% 
         {
            opacity: 1;
            -webkit-transform: translateX(0) rotateY(0deg);
         }
          100% {}
      }
      

The following figure shows the full Tizen logo animation with the transform properties.

Figure: Full Tizen logo animation

Full Tizen logo animation

Source Code

For the complete source code related to this use case, see the following files:

Creating Fade Animation Effects

To enhance the user experience of your application, you must learn to use fade animation effects.

The modal layer pop-up can be used to, for example, show enlarged thumbnail images or notice messages. The modal layer pop-up has the following basic properties:

  • If an event is fired, it gradually becomes visible. This is known as the Fade In effect.
  • The existing background is covered with a translucent layer to make the user focus on the pop-up.
  • When the pop-up is closed, it gradually becomes transparent. This is known as the Fade Out effect.

Figure: Fade effect

Fade effect

You can control UI events and change the DOM elements in the following ways:

  • Events can be controlled with JavaScript, and DOM elements can be devised with CSS.
  • You can use JavaScript frameworks, such as jQuery, Prototype, and Dojo.

To create fade effects:

  • Create a modal layer pop-up using jQuery:
    (function($) 
    {
       function showModalPopup(url) 
       {
          $('body').append('<div class="mask"></div>'); /* Mask in body appended */
          $('.mask').css({'height': $(window).height()}); /* Mask area set */
    
          /* Target layer position set */
          $(url).css(
          {
             'top': ($(window).height()/2 + $(document).scrollTop() - $(url).height()/2) + 'px', 
             'left': ($(window).width()/2 + $(document).scrollLeft() - $(url).width()/2) + 'px'
          });
    
          /* Fade effect */
          $('.mask').fadeTo('slow', 0.7);
          $(url).fadeTo('slow', 1);
       };
    
       $('.layerpopupActive a').on('click', function()
       {
          var targetUrl = $(this).attr('href');
    
          showModalPopup(targetUrl);
    
          return false;
       });
      
       /* End processing - Fade Out effect */
        
       $('body').on('click', function(e) 
       {
          if (e.target.className === 'mask') 
          {
             $('.layerpop').fadeOut();
             $('.mask').fadeOut(400, function() 
             {
                $('.mask').remove();
             });
          };
       });
    })(jQuery);

    If the animation effect is created by using JavaScript, the browser executes the following tasks:

    1. A structure layer is created and added to the document. This is a CPU task.
    2. The added layer is painted as a default value. This is a GPU task.
    3. The layer is painted once again according to the change of value. This is a CPU operation.

    Steps a and c incur CPU tasks, which affect performance the most.

    In case of step a, only 2 layers are created, but as the number of layers created increases, the efficiency of page rendering work drops. In case of step c as well, the more steps it undergoes, the slower the rendering becomes.

    In certain browsers, even if the style of just 1 layer is changed, the entire document is repainted. As the repainting takes only a moment, any animation effects that are supposed to happen cannot be executed in such a short time. This issue occurs frequently in Android™ with severe fragmentation.

    Note
    If a new layer is created in Android 4.0X version, and if the fade effect is used, it only shows the initial value and the result value of the layer style.

    To avoid such a situation to the maximum possible extent, CPU tasks must be reduced to the minimum, and it is better to use transition or animation that uses the internal timer of the browser.

    Note
    In case of using 3D effects, -webkit-transform: translateZ(0); can be used to accelerate the hardware. However, since hardware acceleration support varies between the OS and devices, the actual resulting effects can vary too. Moreover, in the case of version Android™ 2.1, iOS™3.X and below, note that transition and animation may not be realized.
  • Create a modal layer pop-up using CSS3:
    <!--CSS-->
    .mask 
    {
       opacity: 0;
       z-index: -1;
       -webkit-transition: all 400ms ease-in-out;
    }
    
    .mask.active 
    {
       opacity: .7;
       z-index: 498;
    }
    
    <!--Layer pop-up public styles-->
    .layerpop 
    {
       opacity: 0;
       z-index: -1;
       -webkit-transition: all 300ms ease-in-out;
    }
    
    .layerpop.active 
    {
       opacity: 1;
       z-index: 499;
    }
    /* JavaScript */
    function showModalPopup(url) 
    {
       url.className += 'active';
    }

Performance Differences

There can be a difference in performance depending on how the modal layer pop-up is used.

The following figures illustrate the difference in event performance, when using JavaScript and CSS3.

Figure: Event performance using JavaScript

Events using JavaScript

Figure: Event performance using CSS3

Events using CSS3

The following figures illustrate the difference in memory performance, when using JavaScript and CSS3.

Figure: Memory performance using JavaScript

Memory when using JavaScript

Figure: Memory performance using CSS3

Memory when using CSS3

When the styles are applied using JavaScript, the UI thread is used to create the pop-up. The UI thread increases the usage of CPU memory in proportion to the number of pop-ups used.

Note
Tizen provides remote debugging through the JavaScript Debugger tool.
Note

If CSS is used, the handling of JavaScript events and UI DOM operations becomes unnecessary, so the load of the UI thread decreases, and the usage of CPU memory decreases as well. However, CSS3 transitions cannot be applied to versions below Android™ 2.3 and for versions above iOS™ 4.3, which are still widely used, so attention is needed when distributing it to various devices.

Using Hardware Acceleration

The rendering performance of a Web application depends on both the Tizen platform and application design.

In hardware acceleration, GPU is used to perform a function faster than is possible if the application is running on the CPU. It enhances the rendering performance in the dynamic objects used in Web applications.

Using CSS Transition and 3D Transform

To improve the rendering performance, separate moving elements to independent layers as much as possible. You can use CSS transition with the -webkit-transition CSS attribute, or 3D transform with the -webkit-transform attribute. For the best performance, set the -webkit-transform attribute to the 3D type.

In all the following examples, a blue box moves from top left to bottom right for a second.

Figure: Blue box

Blue box

  • Use CPU painting:

    JavaScript performs CPU painting for a moving element for each frame at 16 ms interval using the setTimeout() method over the changing top-left coordinate. This approach does not use hardware acceleration, but only running on the CPU.

    1. Construct the blue box and set its position on the screen:
      <!DOCTYPE html>
      <html xmlns="http://www.w3.org/1999/xhtml">
         <head>
            <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
            <title>JavaScript transition sample</title>
            <style>
               #box 
               {
                  position: absolute;
                  width: 100px;
                  height: 100px;
                  background-color: blue;
                  left: 0px;
                  top: 50px;
               }
            </style>
    2. Perform CPU painting for a moving element at a rate of 1 frame per 16 ms (62.5 frames per second) using the setTimeout() method:

            <script>
               var delta = 0;
               function startTransition() 
               {
                  process();
               }
               function process ()
               {
                  document.getElementById('box').style.left = delta + "px";
                  document.getElementById('box').style.top = delta + 50 + "px";
                  delta += 4;
                  if (delta <= 200)
                     setTimeout(function() {process();}, 16);
               }
            </script>
         </head>
         <body>
            <div id='box' onclick='startTransition()'>click me!</div>
         </body>
      </html>
  • Use CSS transition:

    Separate a moving element to an independent layer with the -webkit-transition CSS attribute. This approach uses hardware acceleration, and can enhance performance while an element is moving.

    <!DOCTYPE html>
    <html xmlns="http://www.w3.org/1999/xhtml">
       <head>
          <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
          <title>-webkit-transition sample</title>
          <style>
             #box 
             {
                position: absolute;
                width: 100px;
                height: 100px;
                left: 0px;
                top: 50px;
                background-color: blue;
                -webkit-transition-duration: 1s;
                -webkit-transition-timing-function: linear;
             }
          </style>
          <script>
             function startTransition()
             {
                document.getElementById('box').style.webkitTransform = "translate(200px, 200px)";
             }
          </script>
       </head>
       <body>
          <div id='box' onclick='startTransition()'>click me!</div>
       </body>
    </html>
  • Use 3D transform:

    Use the -webkit-transform: translate3d 3D transform attribute. The element is separated to an independent layer and uses hardware acceleration irrespective of its movement:

    <!DOCTYPE html>
    <html xmlns="http://www.w3.org/1999/xhtml">
       <head>
          <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
          <title>-webkit-transition with translate3d sample</title>
          <style>
             #box 
             {
                position: absolute;
                width: 100px;
                height: 100px;
                left: 0px;
                top: 50px;
                background-color: blue;
                -webkit-transform: translate3d(0, 0, 0);
                -webkit-transition-duration: 1s;
                -webkit-transition-timing-function: linear;
             }
          </style>
          <script>
             function startTransition() 
             {
                document.getElementById('box').style.webkitTransform = "translate3d(200px, 200px, 0px)";
             }
          </script>
       </head>
       <body>
          <div id='box' onclick='startTransition()'>click me!<p></div>
       </body>
    </html>

Using Accelerated Overflow Scroll

If a page has an overflow scroll, use the -webkit-overflow-scrolling attribute. It separates overflow scroll to an independent layer and enhances rendering performance:

#scroll_area 
{
   overflow: scroll;
   -webkit-overflow-scrolling: touch;
}
Go to top