브라우저 CSS Animation의 동작 과정
date
Sep 16, 2023
slug
browser-animation
summary
Google web dev, Animation 파트를 보고 간략히 정리
thumbnail
status
publish
왜 애니메이션이 느리게 느껴질까...🐢
최신 브라우저들은
transform
& opacity
같은 CSS 프로퍼티들을 이용한 애니메이션을 빠르게 수행하지만그 외의 프로퍼티들을 이용한 애니메이션은 매끄럽게, 60FPS로 수행하지는 않는다.
따라서 스크린이 Update & Repaint 되는 각각의 프레임(Frame)으로 동작하는 웹 특성상, 한 프레임에서
16.7ms(0.0167초)
안에 모든 작업이 이루어지지 않으면 사용자는 딜레이 (Jank) 가 생겼다고 느낄수가 있다.1000ms / 60FPS = 약16.7ms(0.0167초)
Rendering pipeline
브라우저에서 요소를 스크린에 표현하기 위해서는
Rendering pipeline
라는 순차적 단계를 거치게 된다.- Style: 요소들에 적용된 스타일들 계산하기
- Layout: 각가의 요소의 위치(x, y) 요소들 설정하기
- Paint: 요소들이 갖고있는 픽셀들을 Layers에 그리기
- Layer: 요소들이 그룹화되어 있는 부분, Photoshop 레이어와 아이디어가 비슷하다.
- Composite: Layers를 합성해서 스크린에 그리기
Layout 애니메이션
Layout
은 위치(x, y)와 사이즈 변화에 영향을 받는다. 하나의 요소의 위치, 사이즈가 변하게 된다면 자연스럽게 다른 요소들의 Layout도 Recalculate
되는 현상이 발생하게 된다.결국 요소들이 많게 될수록Layout
에 시간을 많이 쏟게 됌!!
Paint 애니메이션
요소들이 어떤
Layers
에 있으냐에 따라서 해당 요소가 다른 요소에 의해 다시 Paint 될 수도 있습니다.pipeline 과정중 가장 긴 시간이기도 합니다.
Composite 애니메이션
Composite
는 Paint에서 생성된 Layer들을 픽셀 정보로 바꾸는 레스터화 (Rasterization), 합성하는 단계로 구성됩니다.애니메이션을 실행하기 위해서는 결국 위 단계들을 처음부터 다시 반복해야 하는데
순차적 특성상, Layout만 바뀌어도 Paint, Composite까지 실행되어야 합니다. 😫
사이트에서 각 엔진마다 CSS 프로퍼티들의 렌더링 단계를 볼 수 있다.
Layer
Paint 단계에서 생성된
Layers
을 통해 다른 요소들의 Layout과 상관 없이 자신의 Layer
에서만 다시 Repaint 작업을 할 수 있습니다.이를 수동(강제)으로는
will-change
를 통해서 Layer
생성을 할 수 있습니다. (Force layer creation)body > .sidebar { will-change: transform; }
// 이렇게는 하지 말자.. *, *::before, *::after { will-change: all; }
will-change
, 이름 그대로 이를 사용하기 위해서는 브라우저에게 숨 좀 돌릴 수 있는 시간이 필요합니다. 😮💨.element:hover { will-change: transform; transition: transform 2s; transform: rotate(30deg) scale(1.5); }
위와 같이 있을때 브라우저는 will-change의 존재 여부와 동시에 애니메이션을 실행하게 됩니다.
.element { /* style rules */ transition: transform 1s ease-out; } .element:hover { will-change: transform; } .element:active { transform: rotateY(180deg); }
따라서 위와 같이 사용자가 hover 하고 클릭할 때까지 시간동안 애니메이션이 일어나기전에 브라우저가
will-change
의 존재를 알고 스스로 최적화를 하도록 해줍니다.하지만,
Layer
는 GPU 자원을 사용하므로 아무래도 추가 연산 & 자원이 소모되기 때문에 꼭 필요한 상황에서만 사용해야 합니다! 만약 will-change
를 통한 작업이 꼭 필요한 상황이라면 해당 작업이 끝난 뒤에 해제를 시켜줘야 합니다!계속 사용되는 인터렉션은 유지, 예를 들어 Sidebar
var el = document.getElementById('element'); el.addEventListener('mouseenter', hintBrowser); el.addEventListener('animationEnd', removeHint); function hintBrowser(event) { event.target.style.willChange = 'transform, opacity'; } function removeHint(event) { // 해제 해주자 event.target.style.willChange = ''; }
wiil-change
에 관한 자세한 좋은 아티클은 여기IE
같이 will-change 지원이 안되는 브라우저는transform: translateZ(0)
통해서 생성 가능..그냥IE
는 편히 보내주자 🪦
그래서 애니메이션이 왜 느려지냐고...🧐
CSS 기반 애니메이션과 Web Animation API는
Compositor Thread
에서 발생합니다.하지만 JS, Rendering pipeline등 작업은
Main Thread
에서 실행!따라서
Compositor Thread
를 이용하지 않는 애니메이션들은 다른 작업들과 같이 이루어지는 Main Thread
에서 작동하므로 느려질수 밖에 없습니다. (FPS가 떨어진다. 🔫)결국 대부분의 애니메이션을
Compositor Thread
를 사용하는 transform
, rotate
, opacity
으로 구현하면 딜레이 없이 애니메이션 구현이 가능해집니다.이동관련
transform
의 translate
, rotate
를 사용하자// 멀리 이동하기~ .animate { animation: slide-in 0.7s both; } @keyframes slide-in { 0% { transform: translateY(-1000px); } 100% { transform: translateY(0); } }
// 제자리 돌리기~ .animate { animation: rotate 0.7s ease-in-out both; } @keyframes rotate { 0% { transform: rotate(0); } 100% { transform: rotate(360deg); } }
크기, 가시성관련
transform
의 scale
& opacity
을 사용하자// 크기 스케일 업, 다운 .animate1 { animation: scale 1.5s both; } @keyframes scale { 50% { transform: scale(0.5); } 100% { transform: scale(1); } } .animate2 { animation: opacity 2.5s both; }
// 가시성 @keyframes opacity { 0% { opacity: 1; } 100% { opacity: 0; } }
📝 결론
Layout, Paint를 다시 작동시키지 말자.
Compositor Thread
를 이용한 애니메이션을 사용하면 좋다. 렌더링 Blocking이 안생기니까will-change
를 통해서 Layer를 강제로 만들 수 있다.하지만 Layer는 양날의 검이다 🗡