iOS 응용 (SwiftUI)

[iOS - SwiftUI] 애니메이션 적용 주의사항 (animation, transition, withAnimation)

jake-kim 2024. 10. 11. 01:16

SwiftUI에서 애니메이션 적용 시 겪는 것

  • 뷰에다가 transition을 적용하려고해도 애니메이션이 걸리지 않음
  • transition을 매번 적용하고 싶은데 매번 동작하지 않음

(위와 관련된 개념들을 아래에서 알아볼 예정)

샘플 코드 준비) 버튼을 누르면 카운트가 올라가는 뷰

  • 아직 애니메이션 적용 x
import SwiftUI

struct ContentView: View {
    var body: some View {
        VStack {
            Text("count: \(cnt)")
            
            Button("refresh") {
               cnt += 1
            }
        }
        .padding()
    }
}

#Preview {
    ContentView()
}

  • 애니메이션을 주는 방법은 3가지가 존재
    • .animation()
    • withAnimation {}
    • .transition()

1. animation으로 애니메이션 주기

  • 단순히 .animation(.easeInOut)만 뷰에 붙이면 애니메이션 적용 완료
import SwiftUI

struct ContentView: View {
    var body: some View {
        VStack {
            Text("count: \(cnt)")  
                .animation(.easeInOut) // <- 추가
            
            Button("refresh") {
               cnt += 1
            }
        }
        .padding()
    }
}

결과)

2. withAnimation 으로 애니메이션 주기

  • withAnimation은 뷰에다가 적용하는게 아닌, 데이터를 변경시켜주는 쪽에서 사용
import SwiftUI

struct ContentView: View {
    var body: some View {
        VStack {
            Text("count: \(cnt)")  
            
            Button("refresh") {
                withAnimation {  // <- 추가
                    cnt += 1
                }
            }
        }
        .padding()
    }
}

결과)

  • withAnimation은 구체적인 애니메이션 타입을 정할 수 있고, 이 값은 첫 번째 방법 .animation과 동일한 옵션을 줄 수 있음
    • 둘 다 기능은 같지만, .animation은 뷰에다 직접 적용하는 방법이고, withAnimation은 데이터를 변경시켜 주는 쪽에서 적용해주는 방법
Button("refresh") {
    withAnimation(.easeInOut) {
        cnt += 1
    }
}

3. .transition()으로 애니메이션 주기

  • transition()은 뷰가 삭제될때나 생성될때만 적용됨 (같은 뷰가 다시 그려질 때는 적용안됨)
  • 첫번째 방법 .animation과 같이 뷰에다 직접 적용
struct ContentView: View {
    @State var cnt = 0
    
    var body: some View {
        VStack {
            Text("count: \(cnt)")
                .transition(.scale) // <-
            
            Button("refresh") {
                  cnt += 1
            }
        }
        .padding()
    }
}
  • 하지만 이렇게하면 애니메이션이 아무것도 걸리지 않음

  • transition은 데이터를 변경해주는 쪽에 withAnimation이 같이 수행되어야 동작
struct ContentView: View {
    @State var cnt = 0
    
    var body: some View {
        VStack {
            Text("count: \(cnt)")
                .transition(.scale) // <-
            
            Button("refresh") {
                withAnimation { // <-
                    cnt += 1
                }
            }
        }
        .padding()
    }
}
  • 이래도 애니메이션이 먹지 않음
    • -> transition은 뷰가 생성될때와 삭제될때만 동작하는데, 생성될때 이미 애니메이션이 동작했기 때문에 다시 동작안함
    • 애니메이션이 매번 걸리도록하는 방법은?
    • 뷰에다가 id를 부여하면 id가 바뀔때마다 뷰가 새로 만들어지는데 이 테크닉을 사용
struct ContentView: View {
    @State var cnt = 0
    
    var body: some View {
        VStack {
            Text("count: \(cnt)")
                .transition(.scale)
                .id("id_\(cnt)") // <-
            
            Button("refresh") {
                withAnimation {
                    cnt += 1
                }
            }
        }
        .padding()
    }
}

결과) transtion으로 애니메이션 적용 성공

애니메이션 적용 정리

  • 1. animation: 뷰에다 직접 적용하고 매번 애니메이션이 먹음
  • 2. withAnimation: 데이터를 변경시켜주는 쪽에서 적용
  • 3. transition: 생성되거나 사라질때만 애니메이션이 동작하며, withAnimation과 id를 부여하여 매번 애니메이션이 동작되게끔 처리가 가능