iOS 기본 (SwiftUI)

[iOS - SwiftUI] EmptyView, EquatableView 사용 방법 (equatable(), body 최적화)

jake-kim 2022. 9. 11. 23:56

목차) SwiftUI의 기본 - 목차 링크

EmptyView

https://developer.apple.com/documentation/swiftui/emptyview

  • 이름 그대로 아무것도 가지고 있지 않은 뷰
    • EmptyView()에 frame, foregroundColor 옵션을 주어도 아무런 표시가 남지 않음
struct ContentView: View {
  var body: some View {
    VStack {
      Text("Text1")
      EmptyView()
        .frame(height: 20)
        .foregroundColor(.blue)
      Text("Text2")
      Spacer()
        .frame(height: 20)
      Text("Text3")
    }
  }
}

  • EmptyView의 사용처
    • 아무런 뷰를 사용하고 싶지 않을때 선언
    • Toggle에 label이라는 파라미터에 EmptyView()를 넣어서 label을 사용하지 않는다는 표현
Toggle(isOn: $isOn, label: { EmptyView() } )

EqutableView

https://developer.apple.com/documentation/swiftui/equatableview

  • 존재하는 뷰 인스턴스의 교체 여부를 감지하는데 사용되는 최적화 기술
    • 원래 @State, @ObservableObject와 같은 뷰의 소스를 변경할 때마다 SwiftUI는 body 속성을 실행하여 새로운 뷰를 생성하지만, EqutableView를 사용하면 매번 새로운 뷰를 생성하지 않는 점이 존재
    • SwiftUI는 변경 사항이 있는 경우 새 View를 렌더링
  • EqutableView 사용 방법
    • Equtable을 준수하는 View를 구현
    • 위에서 구현한 뷰를 EqutableView(content:)로 생성하거나, .equtable()을 추가하여 사용 (.equtable() 방법이 선언적이라 더욱 선호)

ex) 아래 코드는 이곳 참고하여 작성

  • Equtable이 아니라면, 값이 변경될때마다 body가 호출
struct ContentView: View {
  @State private var n = 3

  var body: some View {
    VStack {
      NumberParity(number: n)
      
      Button("New Random Number") {
        self.n = Int.random(in: 1...1000)
      }.padding(.top, 80)
      
      Text("\(n)")
    }
  }
}

extension Int {
  var isEven: Bool { return self % 2 == 0 }
  var isOdd: Bool { return self % 2 != 0 }
}

struct NumberParity: View {
  let number: Int
  @State private var flag = false
  
  var body: some View {
    let animation = Animation.linear(duration: 3.0).repeatForever(autoreverses: false)
    print("init 호출")
    return VStack {
      if number.isOdd {
        Text("ODD")
      } else {
        Text("EVEN")
      }
    }
    .foregroundColor(.white)
    .padding(20)
    .background(RoundedRectangle(cornerRadius: 10).fill(self.number.isOdd ? Color.red : Color.green))
    .rotationEffect(self.flag ? Angle(degrees: 0) : Angle(degrees: 360))
    .onAppear { withAnimation(animation) { self.flag.toggle() } }
  }
}
  • print("init 호출")이 계속 불려서 프린트 되는것을 확인 

  • 현재 ODD, EVEN인 경우, 같은 짝수로만 데이터가 변경될 경우 UI 변경은 필요 없으므로 사실상 body 부분이 다시 불리지 않아도 되는 상태
    • 같은 짝수가 들어오거나 같은 홀수가 연속으로 들어올때 body가 불리지 않도록 EqutableView를 사용
    • Equatable을 준수하고, static func == 로 구현
struct NumberParity: View, Equatable {
  let number: Int
  @State private var flag = false
  
  var body: some View {
    let animation = Animation.linear(duration: 3.0).repeatForever(autoreverses: false)
    print("init 호출")
    return VStack {
      if number.isOdd {
        Text("ODD")
      } else {
        Text("EVEN")
      }
    }
    .foregroundColor(.white)
    .padding(20)
    .background(RoundedRectangle(cornerRadius: 10).fill(self.number.isOdd ? Color.red : Color.green))
    .rotationEffect(self.flag ? Angle(degrees: 0) : Angle(degrees: 360))
    .onAppear { withAnimation(animation) { self.flag.toggle() } }
  }
  
  static func == (lhs: NumberParity, rhs: NumberParity) -> Bool {
      lhs.number.isOdd == rhs.number.isOdd
  }
}
  • 사용하는쪽에서는 2가지 방법으로 사용
    • 위에서 구현한 뷰를 EqutableView(content:)로 생성하거나, .equtable()을 추가하여 사용 (.equtable() 방법이 선언적이라 더욱 선호)
      // in body
      // 1번 방법 - EquatableView 사용
//      EquatableView(content: NumberParity(number: n))
      
      // 2번 방법 - .equatable() 호출
      NumberParity(number: n)
          .equatable()
      Button("New Random Number") {
        self.n = Int.random(in: 1...1000)
      }.padding(.top, 80)
      
      Text("\(n)")

결과) 홀수가 연속으로 들어올때 `init 호출`이 출력되지 않음 (= body 부분이 불리지 않음)

 

* 전체 코드: https://github.com/JK0369/ExEmptyView

 

* 참고

https://swiftui-lab.com/equatableview/

https://swiftwithmajid.com/2020/01/22/optimizing-views-in-swiftui-using-equatableview/

https://swiftui-lab.com/equatableview/

https://developer.apple.com/documentation/swiftui/emptyview