iOS 응용 (SwiftUI)
[iOS - SwiftUI] @ViewBuilder 기능
jake-kim
2024. 9. 11. 01:36
@ViewBuilder 기능 알아보기
@ViewBuilder
var myView: some View {
Text("ex")
}
- @ViewBuilder는 @resultBuilder 기능처럼 return키워드와 콤마를 안쓰고 심플하게 반환할 수 있음
- *resultBuilder개념은 이전 포스팅 글 참고
// resultBuilder를 사용한 경우 - 콤마 존재 x
@PersonBuilder
func getPerson() -> [Person] {
Person(name: "jake", age: 20) // return 키워드를 사용하지 않아도 동작
Person(name: "kim", age: 22)
Person(name: "paul", age: 32)
}
- 하지만 @ViewBuilder를 단순히 @resultBuilder로 만든 것으로 오해할 수 있는데 다른 기능도 존재
@ViewBuilder 중요한 기능
- 1) 위에서 알아본 콤마, return 생략
@ViewBuilder
var myView: some View {
Text("ex")
}
- 2) resultBuilder의 기능과 동일하게 여러가지 뷰를 개행으로 구분하여 한꺼번에 제공 가능
// @ViewBuilder를 안쓰면 에러 발생
// error: Function declares an opaque return type, but has no return statements in its body from which to infer an underlying type
var myView: some View {
VStack {
Text("123")
}
if toggle {
Text("toggle is true")
} else {
Text("toggle is false")
}
}
- 3) @ViewBuilder를 사용하지 않으면 여러가지 유형의 뷰 리턴 불가
// Fail
var myView3: some View {
if Bool.random() {
Text("a") // error: Branches have mismatching types 'Text' and 'EmptyView'
} else {
EmptyView()
}
}
// OK
@ViewBuilder
var myView3: some View {
if Bool.random() {
Text("a")
} else {
EmptyView()
}
}
- 4) 기타 기능
- 가끔 some View 유형을 리턴해야하지만 조건에 따라서 어떤 뷰도 안쓰고 싶을때 EmptyView를 리턴해야하는 상황이 있는데, AnyView로 감싸라고 뜰 때, @ViewBuilder를 붙이면 감싸지 않고 EmptyView바로 사용 가능
- if, else과 같은 조건문이 있을 때 @ViewBuilder를 안쓰면 조건에 따라 뷰가 변경되지 않는 버그가 존재
@ViewBuilder
var someView: some View {
if viewModel.isShowSomeView {
VStack {
Spacer(minLength: 32)
HStack {
AView()
}
}
} else {
EmptyView() // AnyView(EmptyView())
}
}
정리
- @ViewBuilder를 사용하면 뷰를 여러가지 return 키워드 없이 선언형으로 작성이 가능
- 따로 프로퍼티나 함수로 뷰를 관리할 땐 @ViewBuilder를 붙여서 관리하는게 용이
- Swift내장된 View 프로토콜도 역시, body프로퍼티가 @ViewBuilder를 따르고 있음
@available(iOS 13.0, macOS 10.15, tvOS 13.0, watchOS 6.0, *)
public protocol View {
@ViewBuilder @MainActor var body: Self.Body { get }
}