궁금한 내용을 검색해보세요!
이 포스팅은 쿠팡 파트너스 활동의 일환으로, 이에 따른 일정액의 수수료를 제공받습니다.
서근 개발노트
티스토리에 팔로잉
SWIFTUI/View layout

SwiftUI : Overlay / Background / Alignment

서근
QUOTE THE DAY

“ 프로그래머의 문제점은 일이 너무 늦어질 때까지, 뭘하고 있는지 절대 물어볼 수 없다는 점이다. ”

- 시모어 크레이 (Seymour Cray)
Written by SeogunSEOGUN

SwiftUI : Overlay / Background / Alignment

 

overlay / Background / Alignment에 대해 알아보겠습니다.

ZStack

overlayBackground를 이용하면 ZStack에서 사용했던것 처럼 중첩된 뷰를 표현하는것이 가능합니다. 앞으로 자주 사용될 수식어이므로 함께 비교해서 알아봅시다. ZStack은 특성상 그 나중에 작성한  green사각형이 먼저 작성한 red사각형보다 뷰 계층에서 상위에 위치합니다.

swift
UNFOLDED
ZStack {
Rectangle()
.fill(Color.red)
.frame(width: 150.0, height: 150.0)
Rectangle()
.fill(Color.green)
.frame(width: 150.0, height: 150.0)
.offset(x: 10.0, y: 10.0)
}

SwiftUI : Overlay / Background / Alignment - ZStack
ZStack

이제 overlayBackground를 사용해 이것과 동일한 결과를 만들어 내는 코드를 구현해 보겠습니다.

Overlay

overlay는 뷰 원본의 공간을 기준으로 그 위에 새로운 뷰를 중첩하여 쌓는 기능을 하는 수식어 입니다.

 

UIKit에서 addsSubview메서드를 사용하는 개념과 같은데, 자식뷰를 추가하면 부모 뷰를 기준으로 프레임의 좌표가 결정되고 그 크기도 영향을 받았던 것을 생각해 보면 됩니다. ZStack으로 구현한 코드를 overlay를 이용해 구현하면 다음과 같이 작성 할 수 있습니다.

swift
UNFOLDED
/*ZStack없이*/
Rectangle()
.fill(Color.red)
.frame(width: 150.0, height: 150.0)
.overlay( //오버레이 안에 직사각형을 또 추가한다.
Rectangle()
.fill(Color.green)
.offset(x: 10, y: 10))

SwiftUI : Overlay / Background / Alignment - Overlay
Overlay

Background

Background수식어는 overlay와 마찬가지로 뷰 원본의 공간을 기준으로 뷰를 중첩하는것은 같지만, 위가 아니라 그 아래 방향으로 쌓아 나간다는 점이 다릅니다.

 

지금까지 텍스트의 배경색을 지정해 줄 때 사용했던 이 수식어가 사실은 배경색을 지정하는 용도가 아니라, 뷰의 하위 계층에 지정한 색을 가진 또 다른 뷰를 추가하는 기능이었던 것입니다.

 

이번엔 Background를 이용해 구현해 보면 다음과 같이 작성할 수 있습니다.

swift
UNFOLDED
Rectangle()
.fill(Color.red)
.frame(width: 150.0, height: 150.0)
.background(Rectangle()
.fill(Color.green)
.offset(x: 20, y:20))

SwiftUI : Overlay / Background / Alignment - Background
background
SwiftUI : Overlay / Background / Alignment - Background

Alignment

두 수식어에는 공통으로 Alignment매개 변수가 있어 추가되는 뷰의 위치를 설정해 줄 수도 있습니다.

이 매게 변수를 이용한 다른 예제를 살펴봅시다.  padding(),alignment: .  )

swift
UNFOLDED
Circle()
.overlay(Image(systemName: "")
.font(.title)
.padding(),alignment: .top)
swift
UNFOLDED
Circle()
.fill(Color.yellow.opacity(0.8)) //투명도설정
.frame(width: 250, height: 250)
//overlay
.overlay(Text("Joysitck")
.font(.largeTitle))
.overlay(Image(systemName: "arrow.up")
.font(.title)
.padding(),alignment: .top)
.overlay(Image(systemName: "arrow.left")
.font(.title)
.padding(),alignment: .leading) //왼쪽
.overlay(Image(systemName: "arrow.up.right.circle.fill")
.font(.title),alignment: .topTrailing) //오른쪽 위
//background
.background(Image(systemName: "arrow.right")
.font(.title)
.padding(),alignment: .trailing)
.background(Image(systemName: "arrow.down")
.font(.title)
.padding(),alignment: .bottom)

SwiftUI : Overlay / Background / Alignment - Alignment

 

위의 방법을 ZStack를 사용하여 그대로 구현이 가능합니다.

 

내부적으로 뷰가 그려지는 방식은 다르더라도, 결과는 똑같은 모습으로 나타납니다.

swift
UNFOLDED
ZStack {
//Circle전에 작성된 스택은 뒤로 가진다.
VStack {
Spacer() //아이콘이 아래 이니까 spacer를 먼저 작성
Image(systemName: "arrow.down")
.font(.title)
.padding()
}
HStack {
Spacer()
Image(systemName: "arrow.right")
.font(.title)
.padding()
}
Circle()
.fill(Color.yellow)
.opacity(0.8)
.frame(width: 250, height: 250 )
Text("Joystic")
.font(.largeTitle)
//Circle 위로 ZStack이 쌓아진다.
ZStack(alignment: .topTrailing) {
Color.clear
Image(systemName: "arrow.up.right.circle.fill")
.font(.title)
}
VStack {
Image(systemName: "arrow.up")
.font(.title)
.padding()
Spacer() //아이콘이 위 이니까 spacer를 나중에 작성
}
HStack {
Image(systemName: "arrow.left").font(.title).padding()
Spacer()
}
} //최상위 ZStack
.frame(width: 250, height: 250)

 

어느 것을 사용해야한다고 제약이 있는것은 아니지만, 일반적으로 다음과 같이 구분해서 생각해 볼 수 있습니다. 절대적인 기준이 아니므로 참고만 하는것이 좋습니다.

요약

.Overlay / Background

보통 수식 대상이 되는 뷰와 직접적인 연관성이 있는 뷰를 추가할 때 사용하며, 새로운 뷰가 더해진다는 점 외에는 다른 뷰에 영향을 주지 않는다. 그 자체로는 뷰가 아닌 단지 수식어로 존재하는 만큼, 전체 화면의 레이아웃을 구성할 때 사용되기보다 UI의 각 부분을 구성하는 개별적인 뷰 객체들을 꾸밀 때 활용한다.

ZStack

상대적으로 직접적인 연관성이 없는 뷰들을 계층 구조로 나열하여 UI를 구성해야 할 때 사용한다. 자식의 뷰의 크기에 따라 ZStack의 크기도 함께 변할 수 있으므로, overlay / Background수식어를 사용했을 때와 달리 특정 콘텐츠의 변경 사항이 다른 뷰에까지 함께 영향을 줄 수 있 다는 것을 유의해야 한다.

 

 

읽어주셔서 감사합니다🤟

 


잘못된 내용이 있으면 언제든 피드백 부탁드립니다.


서근


위처럼 이미지 와 함께 댓글을 작성할 수 있습니다.