당신이 기다려온 완전한 SwiftUI 문서
Apple의 문서를 넘어서십시오. 코드 예제, 제안 된 모범 사례 및 모든보기, 컨트롤, 레이아웃 등에 대한 설명!
이 문서는 Medium 섹션 링크 제한으로 인해 Chrome에서 가장 잘 볼 수 있습니다.
Apple의 SwiftUI 문서 는 SwiftUI 의 많은 기본 사항을 다룹니다. 하지만 여전히 존재하는 그 격차는 어떻습니까? 우리 모두는 WWDC가 2020 년 6 월에 나올 때 문서가 대규모로 업데이트 될 것이라는 것을 알고 있지만 그렇게 오래 기다릴 수는 없습니다!
다음은 SwiftUI의 기존 문서의 모든 페이지에 대해 배운 모든 것입니다. 나는 애플이 제공 한 것을 반복하지 않을 것이지만 그들이 말하지 않은 것을 추가하려고 노력할 것이다. 일관성을 위해 동일한 범주를 사용할 것입니다. 제 자신을 생각하기에는 너무 게으 르기 때문이 아닙니다.
각 카테고리 제목은 Apple 버전에 직접 연결됩니다.
나는 누구도이 글 전체를 읽을 것이라고 기대하지 않습니다. 그렇게한다면 당신의 헌신에 감탄합니다! 나는 Better Programming 편집자들이 그것을 살펴 봐야한다고 생각하며 그들에게 매우 감사합니다.
지금은이 모든 것이 필요하지 않을 수도 있지만 나중에 필요할 수도 있으므로 책갈피에 추가하고 나중에 지식에 공백이있을 때 다시 돌아 오는 것이 도움이 될 것입니다.
저도 지식에 차이가 있기 때문에 가능한 한 자주이 게시물을 업데이트 할 것을 약속합니다. 더 자세히 설명하거나 다루어야한다고 생각되는 영역이 있으면 알려주세요!
목차
VIEWS AND CONTROLS
- SwiftUI 2 changes
- The View protocol
- Text
- Text ViewModifiers
- Standard text modifiers
- TextField
- TextField ViewModifiers
- SecureTextField
- SecureTextField ViewModifiers
- Font
- Image
- SF Symbols
- Button
- ButtonStyle
- NavigationView and NavigationLink
- EditButton
- MenuButton
- PasteButton
- Toggle
- Creating a custom ToggleStyle
- Picker
- DatePicker
- Slider
- Stepper
VIEW LAYOUT AND PRESENTATION
- HStack, VStack, and ZStack
- List, ScrollView, ForEach, and DynamicViewContent
- Identifiable
- Axis
- Form
- Group
- GroupBox
- Section
- Spacer
- Divider
- TabView
- VSplitView and HSplitView
- Alert
- ActionSheet
- EmptyView
- EquatableView
- AnyView
- TupleView
DRAWING AND ANIMATION
- Animation
- Animatable and AnimatableData
- AnimatablePair
- EmptyAnimatableData
- AnimatableModifier
- withAnimation (Implicit Animation)
- AnyTransition
- InsettableShape
- FillStyle
- ShapeStyle
- GeometryEffect
- Angle
- Edge and EdgeInsets
- Rectangle, RoundedRectangle, Circle, Ellipse, and Capsule
- Path
- ScaledShape, RotatedShape, and OffsetShape
- TransformedShape
- Color
- ImagePaint
- Gradients (Linear/Angular/Radial)
- GeometryReader and GeometryProxy
- CoordinateSpace
FRAMEWORK INTEGRATION
- UIHostingController
- UIViewRepresentable
- UIViewControllerRepresentable
- DigitalCrownRotationalSensitivity
STATE AND DATA FLOW
- State
- Binding
- ObservedObject
- EnvironmentObject
- FetchRequest and FetchedResults
- DynamicProperty
- Environment
- PreferenceKey
- LocalizedStringKey
GESTURES
- Gestures
PREVIEWS
- The PreviewProvider protocol
참고 : SwiftUI 2 변경 사항
View 프로토콜
아직 모르는 경우 SwiftUI는 View
프로토콜을 사용하여 재사용 가능한 인터페이스 요소를 만듭니다. 뷰는 그들이 사용하는 수단 값 유형입니다 Struct
대신의 Class
정의.
이것이 실제로 실제로 무엇을 의미합니까?
구조체는 상속을 허용하지 않습니다. 구조체는 View
프로토콜 을 따르지만 View
Apple이 제공 한 기본 클래스에서 상속하지 않습니다 .
이것은 UIView
UIKit의 거의 모든 것이 상속되는. A는 UIView
기본적으로 프레임이 할당되지 않고 UIViewController
서브 클래스 의 서브 뷰로 추가되지 않으면 볼 수 없습니다 .
View
이라고 ContentView
.ContentView
구조체 안에 body라는 변수가 있음을 알 수 있습니다 . 이것은 View
프로토콜 의 유일한 요구 사항이며 some
Swift 5.1에 새로운 키워드를 사용합니다 .당신은 의지 할 수 스택 오버플로 스레드 더 나은 내가 할 수있는 것보다, 어떤이 키워드 수단을 설명하기 :
"이것을"역 "의 일반적인 자리 표시 자로 생각할 수 있습니다. 호출자가 만족하는 일반 제네릭 플레이스 홀더와는 달리 ... 불투명 한 결과 유형은 구현에 의해 충족되는 암시 적 제네릭 플레이스 홀더입니다. 여기서 빼야 할 중요한 점은 반환하는 함수 some P
가 특정 단일 콘크리트의 값을 반환하는 것입니다. 이 부합 함을 선언을하는 입력합니다 P
. "
View
프로토콜 을 준수하는 Apple이 제공하는 예제보기를 살펴 보겠습니다 .
본문
참조 : 텍스트 (2.0에서 업데이트 됨)
SwiftUI Xcode 프로젝트를 만들 때 얻는 예제 프로젝트에는 아마도 뷰에 대한 가장 간단한 빌딩 블록이 포함되어 있으며 Text
.
대부분 String
의 경우이 생성자에를 전달 하면 표시되는 내용이됩니다.
다음은 모든 이니셜 라이저의 몇 가지 예입니다 Text
.
/* | |
Separate file called Localizable.strings | |
"string_key" = "This string is in the default file"; | |
Separate file called Local.strings | |
"string_key" = "This string is in another file"; | |
*/ | |
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var string = "This is an ObservedObject string" | |
} | |
struct ContentView: View { | |
@ObservedObject var data = DataModel.shared | |
let substring: Substring = "This is a substring" | |
let string = "This is a string" | |
var body: some View { | |
VStack { | |
//This is a substring | |
Text(substring) | |
//This is a string | |
Text(string) | |
//This is an ObservedObject string | |
Text(data.string) | |
//string_key | |
Text(verbatim: "string_key") | |
//This string in the default file | |
Text("string_key") | |
//This string is in another file | |
Text("string_key", tableName: "Local", bundle: Bundle.main, comment: "Comment") | |
} | |
} | |
} |
참고 마지막 초기화하는 데 걸리는 하나 LocalizedStringKey
, tableName
, bundle
,과 comment
는 사용하는 별도의 파일이 필요합니다 .strings
파일 확장자를.
이 initializer에 대한 Apple의 문서 에서 언급했듯이 유일한 필수 매개 변수는 키의 문자열입니다. 이러한 다른 매개 변수가 무엇을 요구하는지 알 수 있도록 대부분 장황한 예제를 제공했습니다.
의 기본 tableName
IS Localizable
하는 문자열 파일의 표준 이름. Local
이 매개 변수가 필요한 이유를 보여주기 위해 일부러 내 이름을 지정했습니다 .
번들은 기본적으로 기본 번들이므로이 Bundle.main
경우 전달 은 중복됩니다.
주석은 상황에 맞는 정보를 제공해야하지만이 예제에서는 문자열 Comment
.
Text
들 여기에 포함 된 VStack
몸 변수의 불투명 결과 유형이 하나 개의 유형으로 설정해야하기 때문이다. 즉, 여러 유형이 포함될 수 있으므로 여러 항목으로 설정할 수 없습니다.
VStack
내부에 최대 10 개의 뷰를 포함 할 수 있지만 각 뷰는 , 또는 VStack
일 수 있으며 각각 내부에 10 개의 유형을 가질 수 있습니다.HStack
Group
이에 대한 자세한 내용을 보려면 HStack, VStack 및 ZStack으로 스크롤 하십시오.
텍스트보기 수정 자
모든 뷰와 마찬가지로 텍스트는 ViewModifier 프로토콜을 준수하는 구조체로 수정할 수 있습니다.
다음과 같이 표면 아래에서 일어나는 일을 볼 수 있도록 사용자 지정 수정 자의 예를 살펴 보겠습니다.
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
VStack { | |
Text("Padding") | |
.padding(10) | |
Text("Padding") | |
.modifier(Padding(value: 10)) | |
Text("Padding") | |
.pad(10) | |
} | |
} | |
} | |
struct Padding: ViewModifier { | |
let value: CGFloat | |
func body(content: Content) -> some View { | |
content | |
.padding(value) | |
} | |
} | |
extension View { | |
func pad(_ value: CGFloat) -> some View { | |
self.modifier(Padding(value: value)) | |
} | |
} |
보시다시피 .modifier(YourModifier())
ViewModifier를 호출 하기 위해 추가 하는 것이 가능 하지만 View
확장 을 사용 하고 깨끗한 호출 사이트를 제공하는 것이 훨씬 더 합리적 입니다.
이것이 표준 수정자가 보이는 방식이므로 View
확장을 만들면 수정자가 기본 수정 자와 훨씬 더 비슷해집니다.
ViewModifier
"modifier"라는 단어로 수정자를 시작하고 생성자를 호출하면 불필요한 복잡성이 추가되기 때문에 이것이 없으면 기본값보다 작성하기 쉬운를 만들기가 어려울 것입니다.
표준 텍스트 수정 자
이 예에서는 VStack
글꼴 정렬을 유지하는 주위에 빨간색 테두리를 두었습니다 . 이것은 내부 VStack
의 Text
s가 고정 된 최대 크기를 갖기 때문에 이것의 경계 가 제한 되어 있음을 보여줍니다 .
이것이 없으면 컨테이너가 s 내부 를 수용하도록 확장 되므로 Text
s에 대한 정렬은 효과가 없습니다 .VStack
Text
선행 (왼쪽) 또는 후행 (오른쪽) 가장자리에 맞추려면 선행 또는 후행 가장자리가있을 위치를 정의해야합니다. 이것은 또한 VStack
자체 의 너비를 고정하여 얻을 수 있습니다 .
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
VStack { | |
Group { | |
//Standard fonts | |
Text("Large title") | |
.font(.largeTitle) | |
Text("Title") | |
.font(.title) | |
Text("Headline") | |
.font(.headline) | |
Text("Subheadline") | |
.font(.subheadline) | |
Text("Body") | |
.font(.body) | |
Text("Callout") | |
.font(.callout) | |
Text("Footnote") | |
.font(.footnote) | |
Text("Caption") | |
.font(.caption) | |
} | |
Group { | |
//Font weights | |
Text("Ultra light") | |
.fontWeight(.ultraLight) | |
Text("Thin") | |
.fontWeight(.thin) | |
Text("Light") | |
.fontWeight(.light) | |
Text("Regular") | |
.fontWeight(.regular) | |
Text("Medium") | |
.fontWeight(.medium) | |
Text("Semibold") | |
.fontWeight(.semibold) | |
Text("Bold") | |
.fontWeight(.bold) | |
Text("Heavy") | |
.fontWeight(.heavy) | |
Text("Bold") | |
.fontWeight(.bold) | |
} | |
VStack { | |
//Font alignments | |
Text("Leading") | |
.frame(maxWidth: 100, alignment: .leading) | |
Text("Center") | |
.frame(maxWidth: 100, alignment: .center) | |
Text("Trailing") | |
.frame(maxWidth: 100, alignment: .trailing) | |
Text("These lines are limited") | |
.frame(maxWidth: 100) | |
} | |
.border(Color.red, width: 1) | |
} | |
} | |
} |
이 예제 Text
에서 View
. 이 경우 수신 된 유형이 a임을 보장 할 수만 있기 때문 Text
입니다.
에 해당하는 작업을 수행하는 것이 불가능 greenStrikethrough
또는 redUnderline
만들어 ViewModifier
나 View
이러한 일반적인 가지고 있기 때문에 확장을 View
을하지 않을 것을 Text
.
이제이 사실을 알았 으므로 위에서 언급 한 s Text
를 만드는 중간 단계없이 를 사용자 지정하는 사용자 지정 함수를 만들 수 있습니다 ViewModifier
.
struct ContentView: View { | |
var body: some View { | |
VStack { | |
Text("Foreground colour") | |
.foregroundColor(.red) | |
Text("background colour") | |
.background(Color.red) | |
Group { | |
Text("background colour with padding") | |
.background(Color.red) | |
.padding() | |
} | |
.border(Color.red, width: 1) | |
Text("Bold") | |
.bold() | |
Text("Italics") | |
.italic() | |
Text("Baseline offset") | |
.baselineOffset(20) | |
.border(Color.red, width: 1) | |
Text("Tracking") | |
.tracking(20) | |
Text("Kerning") | |
.kerning(-1) | |
Group { | |
Text("Underline") | |
.underline() | |
Text("Blue Underline") | |
.underline(color: .blue) | |
Text("Custom Underline") | |
.redUnderline() | |
} | |
Group { | |
Text("Strikethrough") | |
.strikethrough() | |
Text("Red Strikethrough") | |
.strikethrough(color: .red) | |
Text("Custom Strikethrough") | |
.greenStrikethrough() | |
} | |
} | |
} | |
} | |
extension Text { | |
func greenStrikethrough() -> some View { | |
return self.strikethrough(color: .green) | |
} | |
func redUnderline() -> some View { | |
return self.underline(true, color: .red) | |
} | |
} |
TextField
사용자가 텍스트를 입력하도록하려면 해당 데이터를 저장하기위한 바인딩이 필요합니다. 이 첫 번째 예제는 State
SwiftUI 구조체에 문자열을 로컬로 저장하고 실제로 사용하거나 영구적으로 저장할 수있는 곳에 저장하지 않는 변수를 사용합니다.
데이터를 유지하고 계산 된 값을 제공 할 수 있으려면 ObservableObject
. 이것은 본질적으로 데이터를 저장할 일반 Swift 파일을 제공하며, 데이터 값을 수정할 수있는 컨트롤을 추가하기 시작하면 매우 유용합니다.
/* Separate file called Localizable.strings | |
"localized_string_key" = "This is a localized string"; | |
*/ | |
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var observedString = "" { | |
didSet { | |
//RUNS SUCCESSFULLY | |
print("Observed string changed to \(observedString)") | |
} | |
} | |
} | |
struct ContentView: View { | |
@ObservedObject var data = DataModel.shared | |
@State var localString = "" { | |
didSet { | |
//DOES NOT RUN | |
print("Local string changed to \(localString)") | |
} | |
} | |
@State var localFloat = Float() | |
let formatter = NumberFormatter() | |
var body: some View { | |
List { | |
TextField("Enter a local string", text: $localString) | |
Text("@State var localString = \(localString)") | |
TextField("Enter a observed string", text: $data.observedString) | |
Text("@Published var observedString = \(data.observedString)") | |
Text("Other initialisers") | |
TextField("localized_string_key", text: $localString, onEditingChanged: { isFirstResponder in | |
}, onCommit: {}) | |
TextField("Placeholder", text: $localString, onEditingChanged: { isFirstResponder in | |
}, onCommit: {}) | |
TextField("", value: $localFloat, formatter: formatter, onEditingChanged: { isFirstResponder in | |
}, onCommit: { | |
}) | |
TextField("Placeholder", value: $localFloat, formatter: formatter, onEditingChanged: { isFirstResponder in | |
}, onCommit: { | |
}) | |
} | |
} | |
} |
didSet
저장 한 문자열 값에 대한 클로저 를 제공 할 필요가 없습니다 didSet
. 실행 시기를 보여주는 예제 만 제공합니다 . 내 DataModel
클래스는 일반 Swift 클래스이므로 didSet
클로저가 실행되고 문자열의 새 값을 인쇄합니다.
그러나 SwiftUI 뷰는 동적으로 생성되는 값 유형이므로 didSet
콜백은 로컬 State
변수 를 수정할 때 콘솔에 아무것도 인쇄하지 않습니다 .
나중에 이니셜 라이저 중 일부가 a Float
대신 a 를 사용하는 방법을 알고 계셨습니까 String
?
이 이니셜 라이저는 모든 유형을 취하지 만 제공된 Formatter
클래스 중 하나를 전달 하거나 직접 작성해야합니다.
내 예에서는 NumberFormatter
숫자가 아닌 문자를 입력 할 수없는를 사용합니다. 이렇게하면 Float
문자 문자열을 의 값 Float
을 저장하는 로 변환 할 수 없기 때문에 앱이 충돌 할 것이라는 걱정없이 쉽게 저장할 수 있습니다 TextField
.
다른 이니셜 라이저에도 두 개의 클로저가 있으며 그 중 첫 번째는 onEditingChanged
.
이 TextField
이니셜 라이저 에 대한 Apple의 문서 에는 클로저 내부의 bool이 나타내는 내용이 언급되어 있지 않지만 테스트에 따르면 TextField
포커스가 주어진 것과 관련이있는 것으로 보입니다 .
UIKit에서 전화 resignFirstResponder
한 적이 UITextField
있습니까?
UITextField
더 이상 포커스가 필요하지 않기 때문에 기본적으로 키보드를 닫습니다 . 어느 시점에서 키보드를 다시 가져올 수 있더라도 UITextField
다시 첫 번째 응답자로 지정하지 않는 한 텍스트가 삽입되지 않습니다 .
모두가 관련이 있음을 UIResponder
추상적 인 인터페이스, 어떤에서 UIView
, UIViewController
그리고 UIKit 상속의 다른 기본적으로 모든.
SwiftUI 이벤트가 동일한 정도로 처리되는 방식은 알 수 없지만 .NET 을 사용한 모든 사람에게 익숙한 첫 번째 응답자 라는 문구 를 사용하고 UITextField
있습니다.
bool in onEditingChanged
을 호출 fieldActive
하거나 명확하게 표시 할 수있는 다른 항목을 호출 할 수 있습니다 .
중요한 것은 당신이 편집 A를 시작할 때이다 TextField
, onEditingChanged
true로 설정되는 부울 호출됩니다. 키보드의 리턴 키를 누르면 onCommit
블록이 호출 된 후로 onEditingChanged
설정된 부울과 함께 호출됩니다 false
.
TextField ViewModifiers
ViewModifiers에 대한 자세한 설명은 Text ViewModifiers를 참조하십시오 .
struct ContentView: View { | |
@State var localString = String() | |
var body: some View { | |
VStack { | |
TextField("Red text here", text: $localString) | |
.foregroundColor(.red) | |
TextField("Red background here", text: $localString) | |
.background(Color.red) | |
TextField("Red border here", text: $localString) | |
.border(Color.red, width: 1) | |
TextField("Padding here", text: $localString) | |
.padding() | |
TextField("RoundedBorderTextFieldStyle here", text: $localString) | |
.textFieldStyle(RoundedBorderTextFieldStyle()) | |
.background(Color.red) | |
Spacer() | |
} | |
} | |
} |
현재에서 자리 표시 자 텍스트의 전경색을 변경할 수 없습니다 TextField
. 이 글을 쓰는 시점 에서 s를 TextField
표시 할 때 기본값이 아닌 다른 키보드 유형을 사용할 수 없습니다 .TextField
List
TextField
s 를 표시하려고 List
하면 서로 겹치는 현상 이 발생했습니다. 다음은 a에 제시되었을 때 잘 작동하는 것처럼 보이는 모든 키보드 유형입니다 VStack
.
import SwiftUI | |
struct ContentView: View { | |
@State var localString = "" | |
var body: some View { | |
VStack { | |
Group { | |
TextField(".keyboardType(.asciiCapable)", text: $localString) | |
.keyboardType(.asciiCapable) | |
TextField(".keyboardType(.asciiCapableNumberPad)", text: $localString) | |
.keyboardType(.asciiCapableNumberPad) | |
TextField(".keyboardType(.decimalPad)", text: $localString) | |
.keyboardType(.decimalPad) | |
TextField(".keyboardType(.default)", text: $localString) | |
.keyboardType(.default) | |
TextField(".keyboardType(.emailAddress)", text: $localString) | |
.keyboardType(.emailAddress) | |
TextField(".keyboardType(.namePhonePad)", text: $localString) | |
.keyboardType(.namePhonePad) | |
TextField(".keyboardType(.numberPad)", text: $localString) | |
.keyboardType(.numberPad) | |
TextField(".keyboardType(.numbersAndPunctuation)", text: $localString) | |
.keyboardType(.numbersAndPunctuation) | |
TextField(".keyboardType(.phonePad)", text: $localString) | |
.keyboardType(.phonePad) | |
TextField(".keyboardType(.twitter)", text: $localString) | |
.keyboardType(.twitter) | |
} | |
Group { | |
TextField(".keyboardType(.URL)", text: $localString) | |
.keyboardType(.URL) | |
TextField(".keyboardType(.webSearch)", text: $localString) | |
.keyboardType(.webSearch) | |
} | |
} | |
} | |
} |
SecureTextField
기본적으로 TextField
위와 동일하며 입력 한 문자를 숨기는 추가 이점이있어 암호에 유용합니다. 와 마찬가지로 TextField
위에서 만이있는 키보드 종류의 다양한 선택할 수 있습니다 numberPad
여기에 표시됩니다.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var password = "" | |
@Published var passcode = "" | |
} | |
struct ContentView: View { | |
@ObservedObject var data = DataModel.shared | |
@State var password = "" | |
@State var passcode = "" | |
var body: some View { | |
VStack { | |
SecureField("Enter a local password", text: $password) | |
Text("You entered: \(password)") | |
SecureField("Enter a local passcode", text: $passcode) | |
.keyboardType(.numberPad) | |
Text("You entered: \(passcode)") | |
SecureField("Enter an ObservedObject password", text: $data.password) | |
Text("You entered: \(data.password)") | |
SecureField("Enter a, ObservedObject passcode", text: $data.passcode) | |
.keyboardType(.numberPad) | |
Text("You entered: \(data.passcode)") | |
} | |
} | |
} |
SecureTextField ViewModifiers
ViewModifiers에 대한 자세한 설명은 Text ViewModifiers를 참조하십시오 .
과 유사하게 TextField
전경색 또는 배경색을 변경하고 테두리를 추가하고 다른 TextFieldStyle
s를 사용할 수 있지만 현재 자리 표시 자 텍스트의 전경색을 변경할 수 없습니다.
폰트
Apple의 문서에 대해 자세히 설명 할 수 없으므로 Font
Apple의 표준 글꼴과 동일한 방식으로 사용자 지정 글꼴을 사용하는 간단한 방법을 제공했습니다.
/* | |
Requires dragging a font file (MyCustomFont.ttf in this example) into your Project Navigator (the left panel). Click the file in the left panel, and tick your app under "Target Membership" in the File Inspector (the right panel). | |
Do not put the font in the Assets.xcassets folder. Create a folder in your info.plist called 'Fonts provided by application', and inside list the string filenames (MyCustomFont.ttf in this example). | |
Otherwise SwiftUI will not recognise your custom font name. | |
*/ | |
import SwiftUI | |
struct FontView: View { | |
var body: some View { | |
VStack { | |
Text("Custom font text") | |
.font(Font.custom("MyCustomFont", size: 20)) | |
Text("Custom font text") | |
.font(.myCustomFont()) | |
Text("Custom font text") | |
.myFont() | |
} | |
} | |
} | |
extension Font { | |
static func myCustomFont() -> Font { | |
return Font.custom("MyCustomFont", size: 20) | |
} | |
} | |
extension Text { | |
func myFont() -> some View { | |
return self.font(.myCustomFont()) | |
} | |
} |
나는 모두 확장을했습니다 방법을 공지 Font
하고 View
. 내가 Font.custom
직접 사용할 때 알 수 있듯이 확장을 사용할 필요가 없습니다 . 이러한 모든 메서드는 동일한 결과를 가져 Text
오므로 어떤 코드가 가장 깔끔한지는 문제입니다.
작성하기 가장 쉬운 방법은 View
확장 기능으로, 함수에 아무것도 전달할 필요가 없습니다.
Font
확장은 표준 애플 글꼴은 예를 들어, 할당 방식에 더 일치한다 .font(.headline)
.
영상
참조 : 이미지 (SwiftUI 2.0에서 업데이트 됨)
SwiftUI의 이미지 는 UIKit보다 훨씬 쉽습니다. 을 만들고 UIImage(named: “Your file name”)
할당 할 필요없이 yourUIImageView.image
, Image
는 Text
.
a를 전달하면 String
해당 이름의 파일로 설정됩니다. 앱을 실행했는데 해당 이름의 파일이없는 경우 다음과 같은 유용한 콘솔 메시지가 표시됩니다.
No image named ‘Your file name’ found in asset catalog for main bundle.
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
List { | |
Image("Your file name") | |
Image("Your file name", bundle: Bundle.main) | |
.resizable() | |
.frame(width: 100) | |
Image("Your file name", label: Text("My image label")) | |
.resizable() | |
.scaledToFit() | |
Image(decorative: "Your file name") | |
.resizable() | |
.scaledToFill() | |
Image(systemName: "gamecontroller") | |
Image(uiImage: UIImage(named: "Your file name")!) | |
Image(decorative: UIImage(named: "Your file name")!.cgImage!, scale: 0.5, orientation: .rightMirrored) | |
} | |
} | |
} |
이미지는 기본적으로 크기를 조정할 수 없습니다.
후속 수정 자에서 크기를 변경하기 전에 .resizable()
수정자를 호출해야합니다 Image
.
scaledToFit
개질제 이미지의 종횡비를 잠그고 화면 너무 큰 않고 할 수있는 최대 크기로 확장한다.
scaledToFill
이후, 가능성 늘리거나 사용 가능한 공간에 맞게 이미지를 축소하는 것입니다, 수정은 또한 당신의 이미지를 조정하지만 가로 세로 비율을 고정하지 않고.
SF 기호
익숙하지 않다면 SF Symbols 는 Apple이 초경량에서 검은 색까지 9 가지 가중치로 제공하는 1500 개 이상의 기호 라이브러리입니다.
이러한 이미지를 이미지에 사용하려면 이미지에 String
전달하는 레이블 을 systemName
. 사용하려는 기호의 시스템 이름을 알 수 있도록 SF Symbols Mac 앱을 다운로드하는 것이 좋습니다.
SF 기호를 사용하면 이러한 무료 기호의 유연성과 접근성으로 인해 향후 몇 년 동안 iOS 생태계를 장악 할 일관된 모양을 앱에 제공 할 수 있습니다.
단추
참조 : 버튼 (2.0에서 업데이트 됨)
A Button
는 자신의 모습이 없습니다. 즉, 을 준수하는 구체적인 유형 인 Button
a 를 제공해야 합니다 .Label
View
가장 확실한 예는 Text
버튼이 무엇을할지에 대한 정보를 제공하는 것입니다. 이 글을 쓰는 시점에서 Apple이 문서에서 지정하는 유일한 사항은 버튼을 만들고 스타일을 지정하는 방법 외에는 운영 체제에 따라 다르게 트리거된다는 것입니다.
iOS에서는 탭하고, tvOS에서는 버튼이 선택 될 때 Enter 키를 누르고, Catalyst가 있거나없는 macOS 앱에서는 Apple이 언급하지 않은 경우 마우스 나 트랙 패드로 클릭합니다.
생성자는 작업을 제공해야합니다. 이것은 빈 중괄호 세트 일 수 있지만, 최소한이 형식으로 있어야합니다.
import SwiftUI | |
struct ContentView: View { | |
func buttonAction() { | |
print("Button function called") | |
} | |
var body: some View { | |
HStack { | |
Button(action:{print("Button pressed")}) { | |
Text("Button") | |
} | |
Button(action: buttonAction) { | |
Text("Button") | |
} | |
} | |
} | |
} |
중괄호에 기능을 지정하면 매우 빠르게 상세하게 표시 될 수있을뿐만 아니라 중괄호없이 ()
호출 연산자 없이 함수 이름을 지정할 수도 있습니다 . 이것은 작업을 변수에 바인딩하는 것이 아닙니다. 즉 $
, .NET Framework와 같은 바인딩을 사용하는 컨트롤에서 찾을 수 있는 연산자가 필요하지 않습니다 Toggle
.
ButtonStyle
일부 컨트롤을 사용하면 ButtonStyle
이 경우에 따르는 것과 같은 기존 스타일을 선택할 수 있습니다 . 이는 또한 SwiftUI Lab의 Custom Styling tutorial에서Button
찾을 수있는 방법에 대한 세부 사항에 대한 사용자 지정 스타일을 만들 수 있음을 의미합니다 .
해당 게시물에 대한 댓글에서 볼 수 있듯이 SliderStyle
현재 존재하지 않습니다 (Apple 웹 사이트에 문서화되어 있음). 버튼의 기존 스타일을 살펴보고 실제로 작동하는지 살펴 보겠습니다.
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
VStack { | |
Button(action:{}) { | |
Text("Button") | |
} | |
.buttonStyle(DefaultButtonStyle()) | |
Button(action:{}) { | |
Text("Button") | |
} | |
.buttonStyle(PlainButtonStyle()) | |
Button(action:{}) { | |
Text("Button") | |
} | |
.buttonStyle(BorderlessButtonStyle()) | |
#if os(macOS) | |
Button(action:{}) { | |
Text("Button") | |
} | |
.buttonStyle(LinkButtonStyle()) | |
Button(action:{}) { | |
Text("Button") | |
} | |
.buttonStyle(BorderedButtonStyle()) | |
#endif | |
} | |
} | |
} |
일부는 MacOS에서만 사용할 수 있습니다.
NavigationView 및 NavigationLink
참고 항목 : NavigationView (2.0에서 업데이트 됨)
에 뷰를 포함 NavigationView
하면 탐색 제목을 설정하고 다른 뷰에 연결할 수 있습니다. 마찬가지로 에는 기본적으로 프로토콜 을 준수하는 모든 구조체 Button
인 a NavigationLink
가 필요 합니다.Label
View
대부분의 경우 이는 Text
또는 Image
일 수 있지만 사용자가 만든 사용자 정의보기 일 수도 있습니다.
View
아이폰의 오른쪽, 각각의 연속에서 귀하의 링크 슬라이드를 대상으로 NavigationLink
같은 방법으로 슬라이드. 초기으로 돌아 가면 View
왼쪽 가장자리에서 스 와이프하거나 탐색 메뉴의 왼쪽 상단에있는 뒤로 버튼을 사용할 수 있습니다.
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
NavigationView { | |
//Example 1: Using a sequence from 0...49 | |
List { | |
ForEach((0...49), id: \.self) { | |
index in | |
NavigationLink(destination: DogView(index: index)) { | |
Image("full\(index)") | |
.resizable() | |
.scaledToFit() | |
} | |
} | |
} | |
} | |
} | |
} | |
struct DogView: View { | |
let index: Int | |
var body : some View { | |
Image("dog\(index)") | |
} | |
} | |
struct ContentView2: View { | |
let array = ["full0", "full1", "full2", "full3", "full4", "full5", "full6", "full7", "full8", "full9", "full10", "full11", "full12", "full13", "full14", "full15", "full16", "full17", "full18", "full19", "full20", "full21", "full22", "full23", "full24", "full25", "full26", "full27", "full28", "full29", "full30", "full31", "full32", "full33", "full34", "full35", "full36", "full37", "full38", "full39", "full40", "full41", "full42", "full43", "full44", "full45", "full46", "full47", "full48", "full49"] | |
var body: some View { | |
NavigationView { | |
//Example 2: Using an array | |
List(array, id: \.self) { imageName in | |
NavigationLink(destination: DogView(index: Int(self.array.firstIndex(of: imageName) ?? 0))) { | |
Image(imageName) | |
.resizable() | |
.scaledToFit() | |
} | |
} | |
} | |
} | |
} |
이 예제는 내 시계 앱 Dog HQ에서 가져온 것입니다. 전체 크기의 강아지 사진 스크롤 목록을 표시하며 각 사진은 확대 된 버전으로 연결됩니다.
이것이 DogView
내가 어떤 개가 목적지가되고 싶은지 알 수 있도록 인덱스를 확대 된 생성자에게 전달해야하는 이유 입니다.
List
세로로 스크롤하고 원하는 크기로 확장하는를 결합하면 ForEach
50 개의 행을 만들고 지정한 이름으로 해당 인덱스를 클로저에 전달할 수 있습니다.
에 대한 반복 ForEach
은 배열의 최대 항목 수를 갖는 시퀀스 일 수 있거나 배열이 a에 대한 생성자에 전달되고 List
지정한 이름으로 클로저 내부에서 액세스 될 수 있습니다 .
분명히 저는이 배열로 표면을 긁고 있습니다. 배열에는라는 문자열 속성이있는 사용자 정의 클래스 imageName
, 숫자 값 또는 점 구문을 사용하여 액세스 할 수있는 다른 클래스의 인스턴스 와 같은 복잡한 유형이 포함될 수 있습니다 .
화면 상단의 탐색 모음에는 선행 및 후행 단추가 포함될 수 있습니다. 이것의 주요 용도 EditButton
는 아래에서 자세히 설명 하는를 추가하는 것 같습니다.
EditButton
편집 버튼은 List
항목 이 여러 개 있고 일부 항목을 삭제할 수 있도록하려는 경우 매우 유용 합니다. 탭하면 편집 모드 (당연히)로 이동하여 각 행에 수평선이있는 빨간색 원이 표시됩니다.
누르면 편집하는 것은 최종 확인의 역할을 오른쪽 끝의 삭제 버튼을 공개, 왼쪽에 행을 밀어 것입니다.
목록에서 데이터 삭제를 처리하는 함수를 구현해야합니다. 그렇지 않으면 변경 사항이 시각적으로 만 표시되고 데이터가 실제로 예상대로 삭제되지 않습니다.
사용에 대한 자세한 내용 EditButton
은 List, ScrollView, ForEach 및 DynamicViewContent를 참조하십시오 .
MenuButton
MenuButton
macOS 앱에서만 사용할 수 있으므로 모든 표준 .menuButtonStyle
옵션 을 사용하는 Mac 앱 예제를 제공했습니다 .
왼쪽에서 오른쪽으로, 이러한 스타일은 BorderlessButtonMenuButtonStyle
, BorderlessPullDownMenuButtonStyle
하고 PullDownMenuButtonStyle
.
맨 오른쪽에있는 것이 옆에있는 것과 매우 비슷해 보이면 DefaultMenuButtonStyle
.
기본값 MenuButton
은 모양 이므로 PullDownMenuButtonStyle
똑같아 보입니다.
//Note that this example will only work in a Mac app project, as MenuButton cannot be used on iOS | |
import SwiftUI | |
struct ContentView : View { | |
@State var selectedOption = "Select an option" | |
var body : some View { | |
HStack { | |
MenuButton(selectedOption) { | |
Button(action: {self.selectedOption = "Option 1"}) { | |
Text("Option 1") | |
} | |
Button(action: {self.selectedOption = "Option 2"}) { | |
Text("Option 2") | |
} | |
Button(action: {self.selectedOption = "Option 3"}) { | |
Text("Option 3") | |
} | |
} | |
.menuButtonStyle(BorderlessButtonMenuButtonStyle()) | |
MenuButton(selectedOption) { | |
Button(action: {self.selectedOption = "Option 1"}) { | |
Text("Option 1") | |
} | |
Button(action: {self.selectedOption = "Option 2"}) { | |
Text("Option 2") | |
} | |
Button(action: {self.selectedOption = "Option 3"}) { | |
Text("Option 3") | |
} | |
} | |
.menuButtonStyle(BorderlessPullDownMenuButtonStyle()) | |
MenuButton(selectedOption) { | |
Button(action: {self.selectedOption = "Option 1"}) { | |
Text("Option 1") | |
} | |
Button(action: {self.selectedOption = "Option 2"}) { | |
Text("Option 2") | |
} | |
Button(action: {self.selectedOption = "Option 3"}) { | |
Text("Option 3") | |
} | |
} | |
.menuButtonStyle(PullDownMenuButtonStyle()) | |
MenuButton(selectedOption) { | |
Button(action: {self.selectedOption = "Option 1"}) { | |
Text("Option 1") | |
} | |
Button(action: {self.selectedOption = "Option 2"}) { | |
Text("Option 2") | |
} | |
Button(action: {self.selectedOption = "Option 3"}) { | |
Text("Option 3") | |
} | |
} | |
.menuButtonStyle(DefaultMenuButtonStyle()) | |
} | |
.padding() | |
.frame(height: 50) | |
} | |
} |
PasteButton
참고 항목 : PasteButton (2.0에서 업데이트 됨)
이 컨트롤을 사용하면 MacOS에 정보를 붙여 넣을 수 있지만 iOS에서는 사용할 수 없습니다. UTI 유형으로 표현되는 다양한 데이터 유형을 취할 수 있습니다.
이 버튼을 구현할 때 도움이 될 모든 유형의 UTI 문자열을 찾을 수있는 함수를 예제에 포함했습니다. 필요한 유형 식별자를 결정했으면에서 가져온 데이터를 처리해야합니다 NSItemProvider
.
배열의 첫 번째 항목 만 붙여 넣는 예를 보여 드렸지만 다른 데이터 유형과 여러 항목을 처리 할 수있는 방법이 명확 해지기를 바랍니다.
import SwiftUI | |
struct ContentView: View { | |
@State var text = String() | |
var body: some View { | |
VStack { | |
Text(text) | |
PasteButton(supportedTypes: ["public.utf8-plain-text"], payloadAction: { array in | |
array.first!.loadDataRepresentation(forTypeIdentifier: "public.utf8-plain-text", completionHandler: { | |
(data, error) in | |
guard let data = data else { | |
return | |
} | |
let loadedText = String(decoding: data, as: UTF8.self) | |
self.text = loadedText | |
//This call just shows how to find print the UTI type of any type conforming to NSItemProviderWriting, "public.utf8-plain-text" in this case | |
self.getUTITypeString(for: loadedText) | |
}) | |
}) | |
} | |
.frame(width: 200, height: 200) | |
} | |
func getUTITypeString(for item: Any) { | |
if let item = item as? NSItemProviderWriting { | |
let provider = NSItemProvider(object: item) | |
print(provider) | |
} | |
else { | |
print("This data type cannot be used in an NSItemProvider") | |
} | |
} | |
} |
다음 NSItemProviderWriting
은에 따라 붙여 넣을 수 있는 형식 목록입니다 PasteButton
.
CNContact
CNMutableContact
CSLocalizedString
MKMapItem
NSAttributedString
NSMutableString
NSString
NSTextStorage
NSURL
NSUserActivity
UIColor
UIImage
비녀장
참조 : 토글 (2.0에서 업데이트 됨)
Toggle
UISwitch
UIKit 의 SwiftUI에 해당 합니다. IBAction
Swift 코드를 UISwitch
Storyboard에 연결 하고 값이 변경 될 때 실행 되는 함수 대신 SwiftUI는 바인딩을 사용합니다.
변수를 State
(구조체 내) 또는 Published
(에 부합하는 외부 클래스에서 ObservableObject
)로 표시하지 않으면 SwiftUI는 View
값이 변경 될 때의 내용을 다시 그리지 않습니다 .
이것은 Published
SwiftUI가 해당 변수의 존재를 인식하는 유일한 방법이기 때문에 바인딩 프로세스의 필수적인 부분입니다. 특히 외부 코드를으로 표시 합니다.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var toggleOn = false | |
} | |
struct ContentView: View { | |
@State var toggleOn = false | |
@ObservedObject var data = DataModel.shared | |
var body: some View { | |
VStack { | |
Toggle(isOn: $toggleOn) { | |
Text("@State Toggle: \(String(toggleOn))") | |
} | |
Toggle(isOn: $data.toggleOn) { | |
Text("@Published Toggle: \(String(data.toggleOn))") | |
} | |
} | |
.padding() | |
} | |
} |
사용자 지정 ToggleStyle 만들기
에 대한 이니셜 라이저 Toggle
가라는 구조체를 취할 수 있다는 것을 알아 차렸고 ToggleStyleConfiguration
,이를 직접 구성하는 방법을 알아 내려고 잠시 시간을 보냈습니다.
내가 찾은 것은 SwiftUI Lab의 커스텀 스타일링에 대한 훌륭한 튜토리얼의 많은 도움으로 프로토콜이 ToggleStyle
자신 만의 커스텀 스타일을 만드는 기능을 제공한다는 것입니다.
이를 수행 할 수있는 방법 중 일부는 다음 행입니다.
typealias ToggleStyle.Configuration = ToggleStyleConfiguration
typealias
here를 사용하는 것은보다 간결한 로컬 이름으로 구조체를 참조하는 방법 일뿐입니다. 이것은 너무 아마도 makeBody
아래와 기능, 유사한 프로토콜과 같은 선언 서명을 할 수 있습니다 ButtonStyle
, PickerStyle
그리고 TextFieldStyle
:
func makeBody(configuration: Self.Configuration) -> some View
대신 전달 된 레이블을 완전히 무시하고 토글 isOn
상태 에 따라 변경되는 두 개의 동적 레이블을 제공하여 레이블 처리 방법을 변경하기로 결정했습니다 .
import SwiftUI | |
struct ContentView : View { | |
@State var toggleIsOn = false | |
var body: some View { | |
Toggle(isOn: $toggleIsOn) { | |
Text("This label will never be seen") | |
} | |
.toggleStyle(MyToggleStyle(stringWhenOff: "Disabled", stringWhenOn: "Enabled")) | |
Toggle(isOn: $toggleIsOn) { | |
Text("This label will never be seen") | |
} | |
.myToggleStyle(off: "Off", on: "On") | |
} | |
} | |
extension Toggle { | |
func myToggleStyle(off: String, on: String) -> some View { | |
self.toggleStyle(MyToggleStyle(stringWhenOff: off, stringWhenOn: on)) | |
} | |
} | |
struct MyToggleStyle: ToggleStyle { | |
let stringWhenOff: String | |
let stringWhenOn: String | |
func makeBody(configuration: Self.Configuration) -> some View { | |
if configuration.isOn { | |
return Toggle(isOn: configuration.$isOn, label: { | |
Text(stringWhenOn) | |
}) | |
} | |
else { | |
return Toggle(isOn: configuration.$isOn, label: { | |
Text(stringWhenOff) | |
}) | |
} | |
} | |
} |
여기에는 두 가지 예가 있지만 정확히 동일하게 보입니다. 하나는 .toggleStyle
표준과 마찬가지로 수정자를 사용하여 다소 장황한 형식을 사용합니다 ToggleStyle
.
다른 하나는 Toggle
이 장황한 형식을 반환 하는 확장을 사용하여 깔끔한 호출 사이트를 제공하지만 표준 ToggleStyle
모양 과 일치하지 않게됩니다.
당신이 선호하는 것은 당신에게 달려 있습니다. MyToggleStyle
구조체에 지역 변수가 필요하지 않다는 것은 말할 필요도 없습니다.이 변수 가 없으면 생성자에 값을 전달할 필요가 없습니다.
사용자 지정 값을 전달하는 방법을 보여주기 위해서만이 작업을 수행했지만 makeBody
함수 의 서명을 변경할 수는 없습니다 .
즉, 매개 변수 makeBody
만 사용할 수 있습니다 Self.Configuration
. 초기화되지 않은 변수로 구조체를 구성함으로써 isOn
바인딩과 함께 생성자 Label
로부터 값을 전달하는 또 다른 방법이 Toggle
있습니다.
MyToggleStyle
우리가 추가 configuration.label
한 가치 인는 사용하지 않습니다 Text(“This label will never be seen”)
. 이 레이블이 Toggle
없이 구성 될 수 있으므로이 레이블을 추가 할 필요는 없지만 사용자 정의 ToggleStyle
가 원하는 것을 숨기는 방법을 지적 할 가치 가있었습니다.
makeBody
반품 이후 some View
원하는 것을 반환 할 수 있습니다. 당신은을 반환 할 수 Text
, Button
, Image
, 심지어 VStack
, 난 당신이 그렇게 할 거라고 왜 아무 생각이 없지만.
이쑤시개
Pickers 의 Hacking With Swift 튜토리얼에서 언급했듯이 Picker
내부 a 의 기본 동작은 Form
옵션을 선택할 수있는 다른 곳으로 이동하는 것입니다.
iOS에서는을 Form
안에 넣어야합니다 NavigationView
. 그렇지 않으면이 탐색이 발생하지 않습니다. 외부의 Form
의가 DefaultPickerStyle
될 것입니다 WheelPickerStyle
.
나는 또한 포함했다 SegmentedPickerStyle
유사한 모양을 가지고있는 UISegmentedControl
의를 UIKit
.
import SwiftUI | |
struct ContentView : View { | |
var options = ["Option 1", "Option 2", "Option 3", "Option 4"] | |
@State private var selectedOption = 0 | |
var body: some View { | |
NavigationView { | |
Form { | |
Picker(selection: $selectedOption, label: Text("Select a choice")) { | |
ForEach(0 ..< options.count) { | |
Text(self.options[$0]) | |
} | |
} | |
.pickerStyle(SegmentedPickerStyle()) | |
Picker(selection: $selectedOption, label: Text("Select a choice") | |
.frame(minWidth: 100)) { | |
ForEach(0 ..< options.count) { | |
Text(self.options[$0]) | |
} | |
} | |
.pickerStyle(WheelPickerStyle()) | |
Picker(selection: $selectedOption, label: Text("Select a choice")) { | |
ForEach(0 ..< options.count) { | |
Text(self.options[$0]) | |
} | |
} | |
.pickerStyle(DefaultPickerStyle()) | |
} | |
} | |
} | |
} |
날짜 선택기
참고 항목 : DatePicker (2.0에서 업데이트 됨)
DatePicker
과 유사 Picker
하지만 모든 스타일이 동일하지는 않습니다. 내부에서 사용하는 경우 Form
는 DatePicker
단지 한 줄을 차지합니다.
위의 스크린 샷에서 볼 수 있듯이, 기본 DatePicker
A의는 Form
레이블과 현재 날짜가 있습니다. 탭하면 a DatePicker
가 아래로 미끄러 져 나옵니다.
DatePicker
이 정확히 같은 체크 아웃 슬라이드 WheelDatePickerStyle
실제로 난 그냥 때 지금 표시처럼 보이는 이유입니다, WheelDatePickerStyle
아래를.
런타임에 Picker
의 형식을 변경하는 방법을 보여주기 위해 다른 날짜 형식을 시도하는 데 사용할 수있는를 추가했습니다 DatePicker
.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var selectedData = Date() | |
@Published var dateFormatter = DateFormatter() | |
@Published var format = DateFormatter.Style.long | |
@Published var selectedOption = 0 { | |
didSet { | |
format = DateFormatter.Style(rawValue: UInt(selectedOption)) ?? .long | |
dateFormatter.dateStyle = format | |
} | |
} | |
init() { | |
dateFormatter.dateStyle = .long | |
} | |
} | |
struct ContentView : View { | |
@ObservedObject var data = DataModel.shared | |
var options = ["none", "short", "medium", "long", "full"] | |
var body: some View { | |
NavigationView { | |
Form { | |
Picker(selection: $data.selectedOption, label: Text("Select a format")) { | |
ForEach(0 ..< 5) { | |
Text(self.options[$0]) | |
} | |
} | |
.pickerStyle(SegmentedPickerStyle()) | |
DatePicker(selection: $data.selectedData, in: ...Date(), displayedComponents: .date) { | |
Text("Select a date") | |
} | |
Text("Date is \(data.selectedData, formatter: data.dateFormatter)") | |
} | |
} | |
} | |
} | |
슬라이더
슬라이더를 사용하면 최소값과 최대 값 사이에서 엄지 손가락 (흰색 원)을 스 와이프 할 수 있습니다. 이것은 UISlider
UIKit에서 와 유사합니다 . 생성 할 때 SwiftUI가 최소값과 최대 값이 무엇인지 알 수 있도록 닫힌 범위를 설정해야합니다.
단계는 임의의 금액으로 설정할 수 있으므로 값이 소수 일 필요가없는 경우 long Float
을 로 변환 Int
할 필요가 없습니다.
또한 슬라이더 위치가 기록되는 정확도의 양을 늘리거나 줄이는 데 도움이되며, 지정한 단계 양을 지나는 소수 자릿수를 제외하여 계산을 더 쉽게 할 수 있습니다.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var sliderValue = Double() | |
} | |
struct ContentView: View { | |
@State var sliderValue = Double() | |
@ObservedObject var data = DataModel.shared | |
var body: some View { | |
VStack { | |
Slider(value: $sliderValue, in: 0...100, step: 0.1) { | |
_ in | |
} | |
Slider(value: $sliderValue, in: 0...100, onEditingChanged: { sliderActive in | |
}) { | |
Text("Slider") | |
} | |
Slider(value: $sliderValue, in: 0...100, onEditingChanged: { sliderActive in | |
}, minimumValueLabel: Text("Min"), maximumValueLabel: Text("Max")) { | |
Text("Slider") | |
} | |
Slider(value: $sliderValue, in: 0...100, step: 0.1) | |
{_ in | |
print("Value changed") | |
} | |
Slider(value: $sliderValue, in: 0...100, step: 0.1, onEditingChanged: { sliderActive in | |
print("Value changed") | |
}) { | |
Text("Slider") | |
} | |
Slider(value: $sliderValue, in: 0...100, step: 0.1, onEditingChanged: { sliderActive in | |
}, minimumValueLabel: Text("Min"), maximumValueLabel: Text("Max")) { | |
Text("Slider") | |
} | |
} | |
.padding() | |
} | |
} |
스테퍼
Stepper
SwiftUI의 A 는 기본적으로 UIStepper
in UIKit
. 연결된 마이너스 및 플러스 버튼으로 구성됩니다.
모든 이니셜 라이저가 값을 저장하기 위해 바인딩 변수를 설정해야하는 것은 아닙니다. 대부분은 .NET Framework의 값을 감소, 증가 또는 편집 할 때 호출되는 클로저를 사용합니다 Stepper
.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var stepperValue = Double() | |
} | |
struct ContentView: View { | |
@State var stepperValue = Double() | |
@ObservedObject var data = DataModel.shared | |
var body: some View { | |
VStack { | |
Stepper("localized_string_key", onIncrement: {}, onDecrement: {}, onEditingChanged: { editingActive in | |
}) | |
Stepper("Stepper", onIncrement: {}, onDecrement: {}, onEditingChanged: { | |
editingActive in | |
}) | |
Stepper("localized_string_key", value: $stepperValue, in: 0...100, step: 1, onEditingChanged: {editingActive in }) | |
Stepper("Stepper", value: $data.stepperValue, in: 0...100, step: 1, onEditingChanged: { editingActive in | |
}) | |
Stepper("localized_string_key", value: $stepperValue, step: 1, onEditingChanged: { editingActive in | |
}) | |
Stepper("Stepper", value: $data.stepperValue, step: 1, onEditingChanged: { editingActive in | |
}) | |
Stepper(onIncrement: {}, onDecrement: {}, onEditingChanged: { | |
editingActive in | |
}) { | |
Text("Stepper label") | |
} | |
Stepper(value: $stepperValue, step: 1, onEditingChanged: { | |
editingActive in | |
}) { | |
Text("Stepper label") | |
} | |
} | |
} | |
} |
레이아웃 및 프레젠테이션보기
HStack, VStack 및 ZStack
항상 세로로 작성되지만 이러한 스택은 자녀를 다른 방향으로 정렬합니다.
VStack
최대 10 명의 자녀 (및 모든 자손)로 전화 화면을 빠르게 채울 수 있으므로 모든 앱에 유용한 시작점입니다.
HStack
사용 가능한 가로 공간을 사용하여 세로 방향 전화 화면에 많은 공간을 허용하지 않을 수있는 자식을 레이아웃합니다. 이것은 Text
a List
(아래 참조) 와 같이 컨트롤 옆에 레이블 을 배치하려는 경우에 유용합니다 .
List, ScrollView, ForEach 및 DynamicViewContent
참고 항목 : 목록 (2.0에서 업데이트 됨)
참고 항목 : ForEach 및 DynamicViewContent (2.0에서 업데이트 됨)
에 대한 예제에서 언급했듯이 NavigationLink
a List
는 동적 행 수를 수용하기 위해 수직으로 확장되는 스크롤보기입니다. UITableView
in UIKit
과 유사 하지만 작업이 없습니다.
List
와 거의 같은 방식으로에 정적 데이터를 추가하여 다른 데이터 VStack
위에 배치 View
하거나 ForEach
.
ForEach
배열과 같은 컬렉션을 반복하고 매번 표준화 된 방식으로 방대한 양의 데이터를 표시 할 수 있습니다.
ScrollView
스크롤이 가능 VStack
하거나 HStack
그 안에 포함되어 있습니다. 기본값은 ScrollView
의 직접적인 아이가 있더라도, 수직 스크롤 ScrollView
입니다 HStack
.
즉, ScrollView(.horizontal)
이 동작을 재정의하려는 경우 사용해야 합니다. 를 사용하는 ForEach
것처럼 계속 사용할 수 List
있지만, VStack
또는 의 추가 레이어를 HStack
사용하면 더 복잡한 방법이됩니다.
VStacks
물론에는의 UITableView
셀 과 모양이 비슷한 행이 없습니다 UIKit
. 예를 들어 List
로 구성된 A 는 구분선없이 서로 겹쳐 Text
쌓 Texts
입니다.
View
이러한 행을 모방하거나 행에 완전히 다른 모양을 제공 하는 사용자 지정을 만들 수 있습니다 .
그러나 List
가로 스크롤이 필요하지 않으면 사용하는 것이 가장 좋습니다 .
List
또한 사용자 정의 행을 지원 ScrollView
하며 a VStack
가 부족한 다른 기능 이 있습니다.
이 때 EditButton
A와 추가 된 View
를 포함하는 List
, 당신은 재 배열 또는에서 삭제 항목 수 있습니다 List
.
이 상황에서 호출되는 메서드가 없으면 행 List
이 사라지지만 그 뒤에있는 데이터는 영향을받지 않습니다. 다음 번에 스 와이프하여 행을 삭제 한 후 앱을 시작하면 기본 데이터가 수정되지 않았기 때문에 해당 행이 반환됩니다.
에서 해킹으로 스위프트의 onDelete
튜토리얼 , 당신은 어떻게 볼 수있는 .onDelete
수정이 작동합니다. 이렇게하면 사용자가 .NET Framework에서 항목을 삭제하기 위해 스 와이프 할 때 실행되는 메서드를 전달할 수 있습니다 List
.
DynamicViewContent
.onDelete
수정 자의 반환 유형 이지만 ForEach
콘텐츠를 업데이트해야 한다는 의미입니다 .
ForEach
다른 View
구조체는 기본 데이터가 변경 될 때 동적으로 변경 될 수 있음을 의미합니다.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var items = [NSString]() { | |
didSet { | |
UserDefaults.standard.set(items, forKey: "items") | |
} | |
} | |
init() { | |
if let savedArray = UserDefaults.standard.array(forKey: "items") as? [NSString] { | |
items = savedArray | |
} | |
else { | |
items = [NSString("Item 1"), NSString("Item 2")] | |
} | |
} | |
} | |
struct AddButtonView : View { | |
let action: () -> Void | |
var body: some View { | |
Button(action: action) { | |
Text("Add new") | |
} | |
} | |
} | |
struct ContentView : View { | |
@ObservedObject var data = DataModel.shared | |
var body: some View { | |
NavigationView { | |
List { | |
AddButtonView(action: insert) | |
ForEach(data.items, id: \.self) { item in | |
Text(String(item)) | |
} | |
.onDelete(perform: delete) | |
.onMove(perform: move) | |
.onInsert(of: ["public.utf8-plain-text"], perform: inserted) | |
} | |
.navigationBarTitle("Items") | |
.navigationBarItems(trailing: EditButton()) | |
} | |
} | |
func insert() { | |
let string = NSString("Item 3") | |
let provider = NSItemProvider(object: string) | |
print(provider) //output: {types = ("public.utf8-plain-text")} | |
data.items.insert(string, at: 1) | |
} | |
func delete(at offsets: IndexSet) { | |
data.items.remove(atOffsets: offsets) | |
} | |
func move(from source: IndexSet, to destination: Int) { | |
print("Item moved from \(source) to \(destination)") | |
} | |
func inserted(index: Int, provider: [NSItemProvider]) { | |
print("Doesn't seem to get called") | |
} | |
} |
내 예에서 볼 수 있듯이, .onMove
꽤 유사하다 .onDelete
. 실제 문제는을 사용하려고 할 때 발생합니다 .onInsert
.
내가 예상했던 방식은 insert()
함수에 있으며이 방법은 이후 버전의 SwiftUI에서 작동 할 수 있습니다.
어떤 이유로 는 문자열 형식의 식별자 .onInsert
배열 UTType
을 사용합니다. 이는 ForEach
의 기본 데이터에 삽입 될 것으로 예상되는 유형을 지정합니다 NSString
. 이 경우에는 .
UTI 유형 식별자를 만드는 방법의 예로서 NSItemProvider
문자열 에서을 만들어 인쇄했습니다. 이것은에 대한 UTI 유형 문자열을 출력 NSString
하며 이것이 제가 onInsert
호출 에서 인용 한 것 입니다.
그럼에도 불구하고 내가 제공 한 메서드는 호출 inserted()
되지 않습니다. 의 기능 .onInsert
이 추가되지 않았 음을 나타냅니다 . 나는 a List
및 a 내부에서만 시도 VStack
했기 때문에 어딘가에서 작동 할 수도 있습니다.
onInsert
온라인 어디에도 예가 없기 때문에 일 해야한다면 알려주세요 .
식별 가능
ForEach
SwiftUI의 루프는 배열의 각 항목이이어야합니다. 즉 Identifiable
, 각 멤버는 고유 한 식별자를 가지고 있습니다.
다음 예에서는 문자열 배열로 시작합니다. 이후 String
따를 Hashable
프로토콜, 같은 고유 한 식별자가 제공 될 필요 없다 \.self
제공한다 hashValue
.
사용자 지정 클래스에서이 프로토콜을 준수하려면 각 인스턴스를 고유하게 식별 hash(into:)
하는 정수로 필수 구성 요소를 결합 하는 함수 를 제공해야 hashValue
합니다.
또한 함수 ==
에서 결합한 동일한 속성을 비교하는 연산자 를 오버로드해야 hash(into:)
합니다.
Apple 문서에서 Hashable 프로토콜에 대해 자세히 알아보십시오 .
를 만들 때 프로토콜을 myUnhashableType
따르지 않습니다 Hashable
.
결과적으로 \.self
두 번째 위의 주석에서 볼 수 있듯이 ID를 사용하면 작동하지 않습니다 ForEach
. ForEach
주석 처리되거나 제거 되지 않는 한 컴파일을 방해하는 오류가 발생합니다 .
그러나 myIdentifiableType
는 루프에서 훨씬 더 쉽게 식별되는 방법이 있습니다. Identifiable
프로토콜은 ID라는 이름의 변수가 존재해야하고, 각 인스턴스에 고유합니다.
이를 위해 새 인스턴스가 생성 될 때마다 범용 고유 식별자를 생성하는 UUID를 사용합니다.
를 ForEach
준수하면 각 인스턴스를 식별하는 데 필요한 것을 정확히 Identifiable
알려 주기 때문에 에서 식별자를 지정할 필요가 없습니다 ForEach
.
import SwiftUI | |
struct ContentView : View { | |
class myUnhashableType { | |
let string = "Hello world!" | |
} | |
class myIdentifiableType: Identifiable { | |
let string = "Hello world!" | |
let id = UUID() | |
} | |
let array = ["Hello", "world!"] | |
let unhashableArray = [myUnhashableType(), myUnhashableType()] | |
let identifiableArray = [myIdentifiableType(), myIdentifiableType()] | |
var body: some View { | |
VStack { | |
//Works correctly | |
ForEach(array, id: \.self) { item in | |
Text(item) | |
} | |
//Generates the following error: | |
//Referencing initializer 'init(_:id:content:)' on 'ForEach' requires that 'ContentView.myType' conform to 'Hashable' | |
ForEach(unhashableArray, id: \.self) { item in | |
Text(item.string) | |
} | |
//Works correctly | |
ForEach(identifiableArray) { item in | |
Text(item.string) | |
} | |
} | |
} | |
} |
중심선
이것은 단순히 케이스 .horizontal
와 .vertical
. 콘텐츠를 정렬 할 수있는 두 방향을 나타내는 데 사용됩니다.
ScrollView
예를 들어 A 에는라는 속성 axes
이 있습니다 Axis.Set
. 이것은 본질적으로 당신이 중 하나를 포함 축을 바꿀 수 있다는 것을 의미 .horizontal
, .vertical
또는 둘 다. 이렇게하면 스크롤 할 수있는 방향이 변경됩니다.
형태
Form
iOS 설정 메뉴와 다르지 않은 인터페이스를 제공합니다. 인터페이스의 일부를 Section
s 로 분리 할 수 있으며 컨트롤은 List
.
그룹
Apple은 이것을 간단히 다음 과 같이 설명합니다 .
"보기 콘텐츠를 그룹화하기위한 어포던스."
a VStack
또는 HStack
would 처럼 레이아웃에 영향을주는 대신 a Group
는 레이아웃을 전혀 변경하지 않습니다. 대신 최대 10 명의 어린이를 한 명의 어린이처럼 대할 수 있습니다. 예를 들어, a VStack
는 10 개의 자식 만 가질 수 있으므로 10 View
초로 제한 됩니다.
그러나 10 명의 자녀가 모두 Group
s 인 경우 각 그룹은 10 명의 자녀를 가질 수 있으며 1 개에 의해 총 100 개의 View
s가 표시됩니다 VStack
.
그것들이 하나로 취급된다는 사실은 View
또한 각각에 대해 이것을 설정 하거나를 같은 레이아웃에 배치하는 대신 .foregroundColor(.red)
또는 .frame(width: 300)
전체 그룹에 수정자를 적용 할 수 있습니다 .View
View
VStack
GroupBox
GroupBox
View
선택적 레이블이있는 s 의 컨테이너이며 macOS에서만 사용할 수 있습니다.
부분
위의 스크린 샷 Form
은 세 개의 Section
s 로 나누어 진 a 를 보여줍니다 .
보시다시피 Form
에는 세 개의 Section
s 사이에 간격이 있으며 더 어두운 배경색을 표시하는 더 얇은 행으로 볼 수 있습니다.
import SwiftUI | |
struct ContentView : View { | |
@State var toggleIsOn = false | |
@State var sliderValue = 0.0 | |
@State var stepperValue = 0 | |
var body: some View { | |
Form { | |
Section { | |
Text("Section 1") | |
Button(action: {}) { | |
Text("Button") | |
} | |
Toggle(isOn: $toggleIsOn) { | |
Text("Toggle") | |
} | |
} | |
Section { | |
Text("Section 2") | |
Button(action: {}) { | |
Text("Button") | |
} | |
Slider(value: $sliderValue, in: 0...100, step: 0.1) | |
} | |
Section { | |
Text("Section 3") | |
Button(action: {}) { | |
Text("Button") | |
} | |
Stepper(value: $sliderValue, in: 0...100, step: 1) { | |
Text("Stepper") | |
} | |
} | |
} | |
} | |
} |
스페이서
위의 스크린 샷에서, 나는 유사점과 차이점 보여준 Rectangle
와 Spacer
.
경우 rectangleShown
에 해당하는 상기 Text
들에서 HStack
측면과로 푸시 Text
가에서의 VStack
상부 및 하부에 가압된다. 는 Rectangle
기본적으로 부모의 높이 크기를 조정되어 HStack
사용 가능한 모든 수직 공간을 차지하기의 높이.
rectangleShown
false로 설정 Rectangle
하면가 사라지지만 Text
s는 움직이지 않습니다.
때문이다 Spacer
무한대의 최대 크기는 동일한 방식으로 작용한다 Rectangle
. 사용 가능한 모든 공간을 차지하도록 부모의 크기를 늘릴 수 있습니다.
그러나 spacerMaxSize
false로 변경 Spacer
하면 Text
s 의 높이로 축소됩니다. 그렇지 않으면 HStack
의 높이 의 기준이 됩니다.
의 Text
s는 기본적으로 자체가 무한대 HStack
이기 때문에 여전히 측면으로 푸시됩니다 .HStack
maxWidth
요약하면, Spacer
s는 기본적으로 부모의 크기로만 증가하며 최대 크기가 무한대가 아닌 한 부모의 크기를 늘리지 않습니다.
Rectangle
s 와 같은 뷰 는 기본적으로 최대 크기가 무한하며 부모와 동일한 최대 크기가 제공되지 않는 한 부모의 크기를 늘립니다.
import SwiftUI | |
struct ContentView : View { | |
@State var rectangleShown = true | |
@State var spacerMaxSize = true | |
var body: some View { | |
VStack { | |
Text("Text") | |
HStack { | |
Text("Text") | |
if rectangleShown { | |
Rectangle() | |
} | |
else if spacerMaxSize { | |
Spacer() | |
.frame(maxWidth: .infinity, maxHeight: .infinity) | |
} | |
else {Spacer()} | |
Text("Text") | |
} | |
Text("Text") | |
} | |
} | |
} |
분할기
A 는 레이아웃에서 s Divider
사이에 선을 넣습니다 View
. 에서는 VStack
수평선이고에서는 HStack
수직선입니다.
.background(Color.red
)을 설정 Dividers
하면 빨간색 구분선이 표시됩니다. 그렇지 않으면 현재 선택한 색 구성표에 따라 기본값으로 설정됩니다.
import SwiftUI | |
struct ContentView : View { | |
var body: some View { | |
VStack { | |
Text("Text") | |
Divider() | |
HStack { | |
Text("Text") | |
Divider() | |
Text("Text") | |
} | |
Divider() | |
.background(Color.red) | |
Text("Text") | |
} | |
} | |
} |
TabView
참고 항목 : TabView (2.0에서 업데이트 됨)
위의 스크린 샷은 Apple의 TabView
예 를 구현 한 것입니다 .
그것에 추가 할 수있는 것이 많지 않습니다.
VSplitView 및 HSplitView
이러한 버전 VStack
및 HStack
사용자가 각 분할 영역의 크기를 변경하는 디바이더를 드래그 할 수 있습니다.
당연히 VSplitView
자식을 수직으로 배치하고 HSplitView
수평으로 배치합니다. 이것은 macOS에서만 사용할 수 있으므로 iOS 또는 tvOS 프로젝트에서 사용할 수 없습니다.
경보
Alert
s는 만들기가 매우 쉽지만 View
예상대로 프로토콜을 따르지 않습니다 . 당신은 유형의 값을 배치 할 수 없습니다 Alert
A의 VStack
또는 다른 곳에서는 당신이 보여 질 것 같아요.
아래에서 경고를 생성하는 세 가지 주요 시나리오의 예를 제공했습니다. 처음 두 개의 경고에 대한 작업을 추가했지만 필수 사항은 아닙니다. 경고 버튼 중 하나에 대해 작업을 수행하거나 둘 다 수행 할 수 있으며 둘 다 수행 할 수 있습니다.
에서 alert1
기본 작업 (이 경우 "OK"라고 함)을 원합니다.이 작업은 수행 할 작업을 확인합니다. 이것은 콘솔에 "당신이 뭔가를 했어요"를 출력합니다.
옆에있는 버튼 Alert.Button.cancel
은 예상 텍스트를 자동으로 제공하고 누를 때 아무 작업도 수행하지 않는 기본값 을 사용하여 여기에서 생성 된 취소 버튼 입니다.
alert2
눌렀을 때 콘솔에 "당신이 뭔가를 삭제하려고했습니다"를 출력하는 것과 매우 유사합니다. 여기서 차이점은 버튼이 유형 Alert.Button.destructive
이라는 것입니다. 즉, 버튼이 빨간색으로 표시되어 영구적이고 잠재적으로 부정적인 변경을 수행하는 작업을 나타냅니다.
alert3
제목과 선택적 메시지를 가질 수 있지만 Alert
.
import SwiftUI | |
struct ContentView: View { | |
@State var alert1Shown = false | |
@State var alert2Shown = false | |
@State var alert3Shown = false | |
let alert1 = Alert(title: Text("Title"), message: Text("Message"), primaryButton: Alert.Button.default(Text("OK")) {print("You did something")}, secondaryButton: .cancel()) | |
let alert2 = Alert(title: Text("Title"), message: Text("Message"), primaryButton: Alert.Button.destructive(Text("Delete")) {print("You deleted something")}, secondaryButton: .cancel()) | |
let alert3 = Alert(title: Text("Title"), message: Text("Message"), dismissButton: Alert.Button.default(Text("Dismiss"))) | |
var body: some View { | |
Form { | |
Button(action: {self.alert1Shown = true}) { | |
Text("Alert 1") | |
} | |
.alert(isPresented: $alert1Shown) {alert1} | |
Button(action: {self.alert2Shown = true}) { | |
Text("Alert 2") | |
} | |
.alert(isPresented: $alert2Shown) {alert2} | |
Button(action: {self.alert3Shown = true}) { | |
Text("Alert 3") | |
} | |
.alert(isPresented: $alert3Shown) {alert3} | |
} | |
} | |
} |
ActionSheet
똑같은 방식으로 작동하므로 Alert
위의 s에 대한 예제를 ActionSheets
.
가장 큰 차이점은 버튼 배열을 사용한다는 것입니다. 즉, 추가 할 수있는 버튼 수에 제한이 없습니다. 이는 Alert
하나 또는 두 개의 버튼 옵션 만 있는와는 대조적 입니다.
import SwiftUI | |
struct ContentView: View { | |
@State var sheet1Shown = false | |
@State var sheet2Shown = false | |
@State var sheet3Shown = false | |
let actionSheet1 = ActionSheet(title: Text("Title"), message: Text("Message"), buttons: [.default(Text("OK")) {print("You did something")}, .cancel()]) | |
let actionSheet2 = ActionSheet(title: Text("Title"), message: Text("Message"), buttons: [.destructive(Text("Delete")) {print("You deleted something")}, .cancel()]) | |
let actionSheet3 = ActionSheet(title: Text("Title"), message: Text("Message"), buttons: [Alert.Button.default(Text("Dismiss"))]) | |
var body: some View { | |
Form { | |
Button(action: {self.sheet1Shown = true}) { | |
Text("Alert 1") | |
} | |
.actionSheet(isPresented: $sheet1Shown) {actionSheet1} | |
Button(action: {self.sheet2Shown = true}) { | |
Text("Alert 2") | |
} | |
.actionSheet(isPresented: $sheet2Shown) {actionSheet2} | |
Button(action: {self.sheet3Shown = true}) { | |
Text("Alert 3") | |
} | |
.actionSheet(isPresented: $sheet3Shown) {actionSheet3} | |
} | |
} | |
} |
EmptyView
EmptyView
상당히 설명적인 이름이 있습니다. View
공간을 차지하지 않는 보이지 않는 것 입니다.
나는 아래 줄의 예는 사이에 특정 대비립니다 EmptyView
와 Spacer
. Spacer
특정 프레임 크기를 지정하고 해당 공간을 채울 EmptyView
수 있지만 프레임 수정자는 무시합니다.
Spacer
기본적으로 사용 가능한 모든 공간을 채울 것이므로이 예제에서는 높이를 20으로 제한해야합니다.
import SwiftUI | |
struct ContentView: View { | |
var body: some View { | |
VStack { | |
Text("This is text") | |
EmptyView() | |
.frame(height: 20) | |
Text("This is text") | |
Spacer() | |
.frame(height: 20) | |
Text("This is text") | |
} | |
} | |
} |
의 가장 유용한 측면 중 하나는 EmptyView
모든 View
구조체 의 본문으로 반환 될 수 있다는 것 입니다. 즉 View
, 본문이 비어 있기 때문에 오류없이 빈 항목 을 만들 수 있습니다 .
EquatableView
SwiftUI Lab에는 제가 할 수있는 것보다 더 잘 설명 하는 훌륭한 튜토리얼이EquatableView
있습니다.
AnyView
View
프로토콜 이므로 View
자체 인스턴스를 만들 수 없습니다 . 즉 [View]
, 유형의 배열은 만들 수 없지만 [AnyView]
.
다음은 유형 배열의 예 [AnyView]
와 ForEach
. 당신의 생성자에 배열 자체를 전달할 수 없습니다 ForEach
와 같은, AnyView
을 준수하지 않는 Hashable
이 필요된다.
대신 첫 번째 인덱스에서 마지막 인덱스로 이동하는 시퀀스를 만들고이 인덱스를 내부 배열의 첨자로 사용했습니다.
import SwiftUI | |
struct ContentView: View { | |
@State var array = [AnyView(Text("This is")),AnyView(Text("an array")), AnyView(Image(systemName: "gamecontroller"))] | |
var body: some View { | |
VStack { | |
ForEach(0..<array.count) { | |
index in self.array[index] | |
} | |
Button(action: { | |
self.array.shuffle() | |
}) { | |
Text("Shuffle") | |
} | |
} | |
} | |
} |
또한 Button
기본 유형이 AnyView
중요하지 않음 을 보여주기 위해 배열을 섞는를 제공했습니다 . 이전이었다 무엇을 Text
가 될 수 Image
및 AnyView
단지 내용을 다시 그립니다.
TupleView
익숙하지 않은 경우 Swift 언어 문서의 튜플에 대한 설명이 있습니다 .
“복합 유형은 Swift 언어 자체에 정의 된 이름이없는 유형입니다. 복합 유형에는 함수 유형과 튜플 유형의 두 가지가 있습니다.
복합 유형에는 명명 된 유형 및 기타 복합 유형이 포함될 수 있습니다. 예를 들어 튜플 유형 (Int, (Int, Int))
에는 두 개의 요소가 포함됩니다. 첫 번째는 명명 된 유형 Int
이고 두 번째는 또 다른 복합 유형 (Int, Int)
입니다. "
여러면에서 a tuple
는 중괄호 안에 본문이없는 구조체와 같습니다. 구조체에 기본값으로 초기화되지 않은 속성이있는 경우 만들 때 이러한 속성을 괄호로 초기화해야합니다.
튜플에는 이니셜 라이저가 없으므로 해당 대괄호는 tuple
등호 로 a에 할당됩니다 . 구조체 속성 초기화와 달리 레이블은 선택 사항입니다.
import SwiftUI | |
struct ContentView: View { | |
let tuple: (Text, Text, Image) = (Text("This is"), Text("a tuple"), Image(systemName: "gamecontroller")) | |
typealias twoTexts = (firstText: Text, secondText: Text) | |
let myTwoTexts: twoTexts = (firstText: Text("This is"), secondText: Text("Another tuple")) | |
var body: some View { | |
TupleView(tuple) | |
} | |
} | |
struct ContentView_Previews: PreviewProvider { | |
static var previews: some View { | |
ContentView() | |
.previewLayout(.fixed(width: 250, height: 250)) | |
} | |
} |
twoTexts
두 값 twoTexts
이 사용되는 경우 원본과 일치해야하는 두 값에 대한 레이블 이있는 새로운 종류의 튜플을 만드는 두 번째 예제를 제공했습니다 .
나는 여러 s의 콘텐츠를 표시한다는 사실에도 불구하고 주로 a를 사용할 필요가 없다는 사실에주의를 끌기 myTwoTexts
위해 body
of에 추가 하지 않았습니다 .ContentView
TupleView
VStack
View
이런 방식으로 유형을 혼합하는 배열을 만들 수 없습니다. Text
또는 의 배열을 만들 수 Image
있지만 View
프로토콜 요구 사항이 있기 때문에 만들 수 없습니다 . 당신이에서 캐스팅 할 수 없기 때문에 혼합 된 형식의 배열을 작성, 유추 Any
에 View
, Text
또는 Image
.
혼합 유형의 배열을 만드는 방법이 있으며 AnyView
.
이를 AnyView
사용하는 혼합 배열 예제를 보려면 위로 스크롤하십시오 .
대부분의 예제와 달리 나는 그것이 어떻게 보이는지 보여주기 ContentView_Previews
위해 제공 TupleView
했습니다. 나는 작은 View
s 로 작업하고 있기 때문에 고정 된 크기를 사용했으며 별도의 미리보기에서 TupleView
모든 표시를 쉽게 View
볼 수 있습니다.
normal을 사용하면를 View
만들고 별도의 미리보기에 표시 할를 Group
지정 해야합니다 View
. 미리보기에 대한 자세한 내용 은이 게시물 끝 부분 에서 PreviewProvider
프로토콜 에 대해 작성한 내용을 참조하십시오 .
그리기 및 애니메이션
생기
다음은 기본 유형 인 Animation
. 이를 사용하려면 View
s를 변경 하고 .animation(.spring())
수정자를 사용하여 애니메이션을 추가하면됩니다.
수행 할 변경 사항을 구체적으로 지정하려면을 참조하십시오 withAnimation
. 사용자 지정 속성을 사용하여 사용자 지정 모양을 만드는 경우 다음과 같이 지정해야합니다 animatableData
(아래 참조).
import SwiftUI | |
struct ContentView : View { | |
@State var width1 = CGFloat(100) | |
@State var width2 = CGFloat(100) | |
@State var width3 = CGFloat(100) | |
@State var width4 = CGFloat(100) | |
@State var width5 = CGFloat(100) | |
var body: some View { | |
GeometryReader { | |
geometry in | |
VStack { | |
Text("default") | |
.frame(width: self.width1, height: geometry.size.height / 6) | |
.background(Color.blue) | |
.cornerRadius(20) | |
Text("easeIn") | |
.frame(width: self.width2 , height: geometry.size.height / 6) | |
.background(Color.purple) | |
.cornerRadius(20) | |
Text("easeInOut") | |
.frame(width: self.width3 , height: geometry.size.height / 6) | |
.background(Color.orange) | |
.cornerRadius(20) | |
Text("easeOut") | |
.frame(width: self.width4 , height: geometry.size.height / 6) | |
.background(Color.green) | |
.cornerRadius(20) | |
Text("easeOut") | |
.frame(width: self.width5 , height: geometry.size.height / 6) | |
.background(Color.red) | |
.cornerRadius(20) | |
Button(action: { | |
withAnimation{self.width1 = geometry.size.width} | |
withAnimation(.easeIn(duration:3)){self.width2 = geometry.size.width} | |
withAnimation(.easeInOut(duration:3)){self.width3 = geometry.size.width} | |
withAnimation(.easeOut(duration:3)){self.width4 = geometry.size.width} | |
withAnimation(.linear(duration:3)){self.width5 = geometry.size.width} | |
}) { | |
Text("tap to animate") | |
} | |
.frame(width: geometry.size.width, height: geometry.size.height / 6) | |
.background(Color.blue) | |
.foregroundColor(.black) | |
.cornerRadius(20) | |
} | |
} | |
} | |
} |
애니메이션 가능 및 애니메이션 가능 데이터
Animatable
사용자 정의 애니메이션을 적용하는 방법을 SwiftUI에 알리기위한 프로토콜입니다 Shape
.
구조체가 프로토콜을 준수한다고 명시 적으로 선언하지 않고 animatableData
SwiftUI에 애니메이션 할 수있는 것을 알려주 는 속성을 선언하여 준수 할 수 있습니다.
내 예에서는 이미 존재하지만 존재 하지 않는 Square
모양을 만들었습니다 .Rectangle
Square
를 준수 Shape
하려면 모양에 path(in:)
기본적으로 모양의 프레임 사각형을 사용 하는 함수 가 있어야 Path
하며 SwiftUI가 모양을 그리는 데 사용할 수 있는을 생성해야합니다 .
import SwiftUI | |
struct Square : Shape { | |
var offset: CGFloat | |
var animatableData: CGFloat { | |
get {return offset} | |
set {offset = newValue} | |
} | |
func path(in rect: CGRect) -> Path { | |
var path = Path() | |
var length = CGFloat() | |
if rect.width <= rect.height { | |
length = rect.width | |
} | |
else { | |
length = rect.height | |
} | |
path.move(to: CGPoint(x: rect.minX, y: rect.minY - offset)) | |
path.addLine(to: CGPoint(x: rect.minX, y: (rect.minY + length) - offset)) | |
path.addLine(to: CGPoint(x: rect.minX + length, y: (rect.minY + length) - offset)) | |
path.addLine(to: CGPoint(x: rect.minX + length, y: rect.minY - offset)) | |
return path | |
} | |
} | |
struct ContentView : View { | |
@State var offset = CGFloat(15) | |
var body: some View { | |
GeometryReader { | |
geometry in | |
ZStack { | |
Square(offset: self.offset) | |
.foregroundColor(.red) | |
.animation(.spring()) | |
VStack { | |
Spacer() | |
Stepper(value: self.$offset, in: -(geometry.size.height / 2)...geometry.size.height / 2, step: 25) { | |
Text("Offset: \(Int(self.offset))") | |
.foregroundColor(.black) | |
} | |
.padding() | |
.background(Color.white) | |
} | |
} | |
} | |
} | |
} |
내가 할 일은 길이가 더 짧은 지 너비 또는 높이를 결정하는 것입니다. 세로 모드의 iPhone에서 이것은 너비입니다.
경로를 그릴 때 사각형을 사용 rect.maxX
하거나 rect.maxY
제공된 직사각형 공간으로 늘리는 대신 양방향에서이 짧은 길이와 같은 모양을 만듭니다 .
Y 방향에서는 사각형이 화면 중앙의 시작 위치에서 위아래로 이동할 수 있도록 오프셋도 적용합니다.
중요한 부분은 변수 animatableData
에 대한 액세스를 제공하는 getter 및 setter와 함께 라는 변수를 제공한다는 것 offset
입니다.
Stepper
들, 25 단계를 그들이 이동을 의미 Square
때마다 25 번호 변경. 이것이 왜 중요한가요? 이것은 애니메이션이 없다면 흔들리는 움직임이 될만큼 충분히 큰 변화입니다.
애니메이션을 비활성화하면 무슨 뜻인지 알 수 있습니다.
를 사용하는 것에 대해 혼란 스러우면 이 게시물에서 이에 대한 정의를GeometryReader
찾을 수 있습니다 .
AnimatablePair
AnimatablePair
Animatable
위에서 언급 한 프로토콜 과 관련이 있으므로 여기서는 기본 사항을 반복하지 않겠습니다. AnimatablePair
두 개 animatableData
를 하나의 값 으로 압축 할 수 있습니다 . 정말 간단합니다.
다음 은에서 와 a를 모두 가질 Animatable
수있는 위 의 예제 버전입니다 .Square
xOffset
yOffset
import SwiftUI | |
struct Square : Shape { | |
var xOffset: CGFloat | |
var yOffset: CGFloat | |
var animatableData: AnimatablePair<CGFloat, CGFloat> { | |
get {return AnimatablePair(xOffset, yOffset)} | |
set { | |
xOffset = newValue.first | |
yOffset = newValue.second | |
} | |
} | |
func path(in rect: CGRect) -> Path { | |
var path = Path() | |
var length = CGFloat() | |
if rect.width <= rect.height { | |
length = rect.width | |
} | |
else { | |
length = rect.height | |
} | |
path.move(to: CGPoint(x: rect.minX - xOffset, y: rect.minY - yOffset)) | |
path.addLine(to: CGPoint(x: rect.minX - xOffset, y: (rect.minY + length) - yOffset)) | |
path.addLine(to: CGPoint(x: (rect.minX + length) - xOffset, y: (rect.minY + length) - yOffset)) | |
path.addLine(to: CGPoint(x: (rect.minX + length) - xOffset, y: rect.minY - yOffset)) | |
return path | |
} | |
} | |
struct ContentView : View { | |
@State var xOffset = CGFloat(15) | |
@State var yOffset = CGFloat(15) | |
var body: some View { | |
GeometryReader { | |
geometry in | |
ZStack { | |
Square(xOffset: self.xOffset, yOffset: self.yOffset) | |
.foregroundColor(.red) | |
.animation(.spring()) | |
VStack { | |
Spacer() | |
Group { | |
Stepper(value: self.$xOffset, in: -(geometry.size.height / 2)...geometry.size.height / 2, step: 25) { | |
Text("X Offset: \(Int(self.xOffset))") | |
.foregroundColor(.black) | |
} | |
.padding() | |
Stepper(value: self.$yOffset, in: -(geometry.size.width / 2)...geometry.size.width / 2, step: 25) { | |
Text("Y Offset: \(Int(self.yOffset))") | |
.foregroundColor(.black) | |
} | |
.padding() | |
} | |
.background(Color.white) | |
} | |
} | |
} | |
} | |
} |
EmptyAnimatableData
위의 AnimatablePair
및 AnimatableData
예제에서 사용자 정의 모양에서 애니메이션 할 것으로 예상되는 속성을 SwiftUI에 명시 적으로 알 렸습니다.
Shape
자체적으로 Animatable
프로토콜을 따르기 때문에 기본 구현이 상속됩니다. 기본 구현은 animatableData
유형으로 설정된 속성을 만드는 것 입니다 EmptyAnimatableData
.
이를 통해의 자식은 실제로를 설정하지 않고도 프로토콜 Shape
을 준수 할 Animatable
수 animatableData
있습니다.
이 값을 재정의하려면 위의 예에서했던 것처럼 할 수 있습니다.
AnimatableModifier
AnimatableModifier
애니메이션이있는 수정 된 뷰를 생성 할 수 있습니다.
대신 Animatable
프로토콜을 사용하는 AnimatableModifier
프로토콜 에 대한 위 섹션에서 예제 버전을 생성했습니다 . 기존 예제에서는 정사각형을 매번 25 씩 위아래로 움직여 움직이는 스테퍼가있었습니다.
이 새 버전에서는 Rectangle
전체 화면을 채우는 파란색 을 사용 하고 AnimatableSquare
수정자를 적용합니다 .
내가 사용하는 참고 View
어색한 사용하지 않는 확장을 .modifier(SquareAnimatable(offset: offset))
일반적으로 사용자 정의 수정에 필요한 구문을.
SquareAnimatable
수정은 빨간색을 추가 Square
어느 오버레이로 View
. 그 오버레이는 그것을 생성하는 offset
에서 그 값을 취 View
합니다. 즉, 부모 View
는 그 값을 변경할 수 SquareAnimatable
있고 사각형을 이동하고 변화를 애니메이션 할 것입니다.
Square
이 예제의 모양은 실제로의 범위 내에서 선언 SquareAnimatable
부모 있음을 의미 수정, View
와 Stepper
는 오프셋 값은 아무 생각이 없습니다 전달 방법도 만들 수 Square
는 제어하다고!
import SwiftUI | |
struct SquareAnimatable : AnimatableModifier { | |
var offset: CGFloat | |
var animatableData: CGFloat { | |
get {return offset} | |
set {offset = newValue} | |
} | |
func body(content: Content) -> some View { | |
return content.overlay( | |
Square() | |
.frame(width: 100, height: 100) | |
.foregroundColor(.red) | |
.offset(x: 0, y: offset) | |
) | |
} | |
struct Square : Shape { | |
func path(in rect: CGRect) -> Path { | |
var path = Path() | |
var length = CGFloat() | |
if rect.width <= rect.height { | |
length = rect.width | |
} | |
else { | |
length = rect.height | |
} | |
path.move(to: CGPoint(x: rect.minX, y: rect.minY)) | |
path.addLine(to: CGPoint(x: rect.minX, y: (rect.minY + length))) | |
path.addLine(to: CGPoint(x: rect.minX + length, y: (rect.minY + length))) | |
path.addLine(to: CGPoint(x: rect.minX + length, y: rect.minY)) | |
return path | |
} | |
} | |
} | |
extension View { | |
func animatableSquare(offset: CGFloat) -> some View { | |
return self.modifier(SquareAnimatable(offset: offset)) | |
} | |
} | |
struct ContentView : View { | |
@State var offset = CGFloat(15) | |
var body: some View { | |
GeometryReader { | |
geometry in | |
ZStack { | |
Rectangle() | |
.foregroundColor(.blue) | |
.animatableSquare(offset: self.offset) | |
.animation(.spring()) | |
VStack { | |
Spacer() | |
Stepper(value: self.$offset, in: -(geometry.size.height / 2)...geometry.size.height / 2, step: 25) { | |
Text("Offset: \(Int(self.offset))") | |
.foregroundColor(.black) | |
} | |
.padding() | |
.background(Color.white) | |
} | |
} | |
} | |
} | |
} |
처음에이 작업을 수행했을 때 오프셋을 @Binding
.
이러지마!
의 setter animatableData
가 전달 된 값을 설정하려고하면 다음과 같은 런타임 경고가 표시됩니다.
Modifying state during view update, this will cause undefined behavior
본질적으로 문제는 오프셋@State
이라는 초기 속성 이 생성되는 동안 변경하려고한다는 것입니다. .NET이라면 문제가되지 않을 것 입니다.View
Button
withAnimation (암시 적 애니메이션)
SwiftUI에 애니메이션을 가져 오는 쉬운 방법 View
은 withAnimation
블록 안에 코드를 배치하는 것 입니다.
이에 사용되는 블록과 유사 UIView.animate(withDuration:Animations:)
에 UIKit
있지만, 기본적으로 기간을 고려하지 않습니다.
애니메이션이 어떻게 보이는지에 대한 더 많은 제어를 추가 하는 Animation
것과 같은 객체를 전달할 수 있습니다 withAnimation(.linear(duration: 5))
. 기간을에 전달할지 여부도 선택할 수 있습니다 .linear
.
에 대한 자세한 내용 은이 섹션의 시작 부분에 Animation
있는 정의를 참조하십시오 .
import SwiftUI | |
struct ContentView : View { | |
@State var height = CGFloat(0) | |
var body: some View { | |
ZStack { | |
VStack { | |
Rectangle() | |
.frame(height: height) | |
} | |
Button(action: {withAnimation {self.height += 25}}) { | |
Text("Tap to animate") | |
} | |
.frame(width: 150, height: 75) | |
.background(Color.blue) | |
.cornerRadius(20) | |
.foregroundColor(.white) | |
.padding() | |
} | |
} | |
} |
에 .animation(.default)
수정자를 추가하면 Rectangle
이 경우 동일한 효과가 있지만 차이점을 지적 할 가치가 있습니다.
.animation(.default)
개질제 암시 애니메이션의 일례이다. 기본적으로에 대한 변경 사항 Rectangle
이 애니메이션 될 것으로 예상한다고 말하는 것입니다 .
즉 Button
, 너비를 늘리는 다른 항목 을 추가 하면 이러한 변경 사항을로 지정하지 않아도 애니메이션이 적용됩니다 withAnimation
.
View
애니메이션 할 수있는의 측면을 완전히 제어 하려면를 사용하여 애니메이션 할 값 변경을 정확하게 지정하여 명시해야합니다 withAnimation
.
AnyTransition
이는 AnyView
뷰를 제네릭 및 불투명 반환 유형으로 처리 할 수있는와 유사 합니다.
여러 전환을 결합 할 때이 combined
방법을 사용하여 전환을 함께 추가 할 수 있습니다 . Animation
전환에을 추가 할 수도 있으며 그 결과는 AnyTransition
객체입니다.
InsettableShape
strokeBorder
, 모양의 영역을 자르는 테두리를 그릴 수 있도록하려면 a Shape
가 InsettableShape
프로토콜을 준수해야합니다 .
자세한 내용은이 Swift를 사용한 해킹 튜토리얼 을 참조하십시오.
FillStyle
FillStyle
두 가지 옵션 만 있으며 둘 다 bool입니다.
짝수 홀수 규칙은 SwiftUI가 경로의 어떤 부분을 채워야 하는지를 결정하는 방법과 관련됩니다. SVG 사양 을 인용하려면 :
"이 규칙은 임의의 방향으로 해당 지점에서 무한대까지 광선을 그리고 광선이 교차하는 지정된 모양의 경로 세그먼트 수를 계산하여 캔버스에서 지점의"내부 "를 결정합니다.
이 숫자가 홀수이면 점이 안쪽입니다. 짝수라면 그 지점은 밖에 있습니다.”
실제로 이로 인해 왜곡 된 채우기가있는 모양이 겹치는 부분에 채우기가 없도록 자체적으로 겹칩니다. SVG 사양과 짝수 홀수 규칙에 대한 Wikipedia 페이지 에이 효과의 예가 있습니다 .
isEOFilled
매개 변수에 대해 false를 사용하면 0이 아닌 방법이 사용됩니다. 이렇게하면 겹치지 않는 공간뿐만 아니라 모든 닫힌 공간이 채워집니다.
안티 앨리어싱은 가장자리의 계단 또는 '부드럽게하는 과정입니다 계단 현상 '. 해상도가 낮 으면 래스터 이미지가 정사각형 픽셀 격자로 구성되어 있기 때문에 재기 스 현상이 발생합니다.
수평 및 수직 직선은 들쭉날쭉하지 않고 저해상도로 렌더링 될 수 있지만, 이러한 축과의 각도 차이로 인해 축 중 하나에 수직이 아닌 선을 나타내려고하는 "계단 효과"로 인해 들쭉날쭉 한 현상이 나타납니다. 정사각형 픽셀 격자.
isAntialiased
매개 변수에 true를 사용하면 약간의 블러 링이 사용되어 들쭉날쭉 한 부분을 부드럽게합니다. 그렇지 않으면 재기 스가 발생합니다.
ShapeStyle
ShapeStyle
View
s에서 Shape
s 를 만드는 데 사용됩니다 . 놀랍게도 현재 문서가없는 배경 수정자는 다음과 같이 선언됩니다.
extension View {
@inlinable public func background<Background>(_ background: Background, alignment: Alignment = .center) -> some View where Background : View
}
실제로 프로토콜을 직접 따르지 않는 사실에도 불구하고 본문으로 View
만 있는를 만들 수 있다는 사실에 놀라실 수 있습니다 .Color
Color
View
대신 Color
에 부합 ShapeStyle
, 어떤을 준수 View
자체.
무슨 일이 일어나고 있는지는 의 기본 구현 에 따라 채우는 Rectangle
으로 생성되는 것입니다.Color
ShapeStyle
extension ShapeStyle { | |
/// Return a new paint value matching `self` except using `rect` to map unit-space coordinates to absolute coordinates. | |
@inlinable public func `in`(_ rect: CGRect) -> some ShapeStyle | |
} | |
/// Default View.body implementation to fill a Rectangle with `self`. | |
extension ShapeStyle where Self : View, Self.Body == _ShapeView<Rectangle, Self> { | |
public var body: _ShapeView<Rectangle, Self> { get } | |
} |
ImagePaint
, Border
, Stroke
, Fill
, 그리고 Gradients
모든 것은 사용하는 것 ShapeStyle
전경과 배경을 만드는 데 어떤 방법으로.
GeometryEffect
GeometryEffect
제공되는 rotation3DEffect
수정 자 와 유사한 3D 효과를 제공하는 사용자 정의 애니메이션을 만들 수 있습니다 .
SwiftUI Lab에는 GeometryEffect 사용에 대한 훌륭한 튜토리얼이 있습니다 .
각도
Angle
도 또는 라디안으로를 만들 수 있으며 둘 다 Double
.
일단 생성되면 각도 또는 라디안을 Angle
속성으로 액세스 할 수 있습니다. Angles
는 RadialGradients
,에 호를 Paths
만들고에 RotationEffect
및에서 RotationGesture
.
Edge 및 EdgeInsets
Edge
값을 포함하는 ENUM이고 .bottom
, .leading
, .top
, 및 .trailing
. 두 가지 수정 자에만 사용되는 것 같습니다 : .edgesIgnoringSafeAreas
및 .padding
.
EdgeInsets
.listRowInsets
및 .resizable
수정 자 와 같은 장소에서 사용됩니다 .
Rectangle, RoundedRectangle, Circle, Ellipse 및 Capsule
이러한 셰이프 Views
는 SwiftUI에서 제공되는 것처럼 쉽게 만들 수 있습니다 . 아래 이미지는 이들의 차이점을 보여줍니다.
Circle
가로 세로 비율을 잠그는 유일한 것으로 보이므로 높이보다 너비가 더 큰 프레임을 제공하더라도 (아래 코드에서 Circle
와 같이) Ellipse
.
라는 해당 항목이 없으므로 Square
a를 사용하고 Rectangle
동일한 너비와 높이를 지정 하여 정사각형을 만들 수 있습니다 .
바라건대, 나란히 비교하면 Capsule
a가 RoundedRectangle
. 나는 15의 RoundedRectangle
a cornerRadius
를 제공 했기 때문에 상단 가장자리가 눈에.니다.
I가 설정된 경우 RoundedRectangle
의를 cornerRadius
이 경우 50의 폭의 50 %로, 상기에서 거의 구별 모양을 갖는다 Capsule
.
요약하면 a Capsule
는 항상 너비의 50 % RoundedRectangle
와 같은 a 와 cornerRadius
같습니다.
A는 Rectangle
또한 동일 RoundedRectangle
A의 cornerRadius
중 0
.
import SwiftUI | |
struct ContentView : View { | |
var body: some View { | |
VStack { | |
HStack { | |
VStack { | |
Text("Circle") | |
Circle() | |
.frame(width: 150, height: 50) | |
} | |
VStack { | |
Text("Ellipse") | |
Ellipse() | |
.frame(width: 150, height: 50) | |
} | |
} | |
HStack{ | |
VStack { | |
Text("Rounded rectangle") | |
RoundedRectangle(cornerRadius: 15) | |
.frame(width: 100, height: 200) | |
} | |
VStack { | |
Text("Capsule") | |
Capsule() | |
.frame(width: 100, height: 200) | |
} | |
VStack { | |
Text("Rectangle") | |
Rectangle() | |
.frame(width: 100, height: 200) | |
} | |
} | |
} | |
} | |
} |
통로
섹션에서 내가 말을 반복합니다 Animatable
및 AnimatableData
:
"을 준수 Shape
하려면 모양 path(in:)
에 기본적으로 모양의 프레임 사각형을 Path
사용하고 모양을 그리는 데 사용할 수있는 SwiftUI 를 생성 해야하는 함수가 있어야합니다 ."
Apple 튜토리얼 드로잉 경로 및 모양 은 .NET FrameworkView
를 준수하지 않고 의 body 속성에서 직접 경로를 사용합니다 Shape
.
ScaledShape, RotatedShape 및 OffsetShape
이러한 변환 Shapes
은 a Shape
와 변환에 필요한 매개 변수를 전달하기 만하면되므로 사용하기가 매우 쉽습니다 .
내 scaled의 Rectangle
경우 양방향으로 0.5만큼 배율을 조정했습니다. A Rectangle
는 일반적으로 사용 가능한 모든 공간을 차지하므로이 공간이 상대적으로 작고 VStack
.
회전 된 사각형과 오프셋 된 사각형은 프레임을 사용하여 축소해야했습니다. 그렇지 않으면 다른 사각형과 겹칠 것입니다.
이 스크린 샷은 라이트 모드에서 촬영되었으므로 기본값 foregroundColor
은 검은 색입니다. Text
처음 2 개의 오버레이 된 s를 흰색으로 변경했지만 다크 모드에서는 작동하지 않습니다.
다크 모드에서 Rectangle
s는 기본값 foregroundColor
인 흰색을 사용합니다. 따라서이 텍스트는 숨겨 Text
지고 오프셋의 Rectangle
경우 기본 검은 색 배경으로 숨겨집니다.
SwiftUI에서 기본 배경 및 전경색을 사용할 때는 항상 밝고 어두운 모드를 고려하십시오.
import SwiftUI | |
struct ContentView : View { | |
var body: some View { | |
VStack { | |
ScaledShape(shape: Rectangle(), scale: CGSize(width: 0.5, height: 0.5)) | |
.overlay(Text("Scaled shape") | |
.foregroundColor(.white)) | |
RotatedShape(shape: Rectangle(), angle: Angle(degrees: 45)) | |
.frame(width: 150, height: 150) | |
.overlay(Text("Rotated shape") | |
.foregroundColor(.white)) | |
Group { | |
OffsetShape(shape: Rectangle(), offset: CGSize(width: 110, height: 50)) | |
.frame(width: 100, height: 100) | |
.overlay(Text("Offset shape")) | |
} | |
.frame(height: 200) | |
} | |
} | |
} |
나는 200에서 자체 고정 높이를 갖는 OffsetShape
예제를 넣었다 Group
.
이는 도형을 오프셋해도 할당 된 공간의 크기가 증가하지 않기 때문입니다. 따라서 할당 된 공간은의 높이로 인해 100이 OffsetShape
되지만 아래로 이동하고 오른쪽으로 이동하면 단순히 OffsetShape
하단의 하단 에서 벗어나는 것 입니다. 화면.
RotatedShape
, 마찬가지로 회전을 설명하기 위해 할당 된 공간을 늘리지 않습니다.
오버레이 Text
는으로 오프셋되지 않으므로가 있어야하는 곳에 남아있는 OffsetShape
곳에 재미있는 효과를 생성 합니다.Text
OffsetShape
TransformedShape
A는 TransformedShape
받는 유사하다 ScaledShape
, RotatedShape
그리고 OffsetShape
예 상술 그것이 단일 매개 변수를 제외하고 CGAffineTransform
는 초기화된다.
Core Graphics Affine Transform은 변환을 3 x 3 행렬로 나타내므로 단일 인스턴스에서 많은 변환을 나타낼 수 있습니다. 3 x 3 행렬의 맨 오른쪽 열에는 항상 [0, 0, 1]이 있으므로 여기에서 변경 한 모든 내용은 처음 2 개에 있습니다.
행렬 곱셈이 어떻게 작동하는지 모른다면 이 zany 웹 사이트를 확인하십시오 .
GAffineTransform
초기화시 또는 나중에 개별 위치 a , b , c , d , t ₓ 및 t ᵧ에 대한 속성을 조정 하여 클래스를 사용하여 행렬을 생성 할 수 있지만, 클래스 를 사용하는 가장 쉬운 방법은 생성자 및 Apple이 제공하는 인스턴스 메소드.
내 예에서는 번역을 수행하는 생성자를 사용 Rectangle
하여 화면 중앙을 중앙에 배치합니다. 그런 다음 크기를 조정하고 45도 회전합니다.
import SwiftUI | |
struct ContentView : View { | |
func complexTransformation() -> CGAffineTransform { | |
var affineTransform = CGAffineTransform(translationX: 250, y: 200) | |
affineTransform = affineTransform.scaledBy(x: 0.4, y: 0.4) | |
affineTransform = affineTransform.rotated(by: 45) | |
return affineTransform | |
} | |
var body: some View { | |
TransformedShape(shape: Rectangle(), transform: complexTransformation()) | |
} | |
} |
인스턴스 메서드의 결과를 변수 자체에 할당해야하므로 어색한 구문을 사용합니다 affineTransform = affineTransform.scaledBy(x: 0.4, y: 0.4)
.
Xcode가 일반적인 방식으로 다음과 같이 경고하지 않기 때문에 원래 메서드를 직접 호출하면 작동한다고 가정했습니다.
Result of call to scaledBy(x:y:) is unused
색깔
SwiftUI에는 Color
크로스 플랫폼 인 자체 클래스가 있습니다. 즉, macOS, tvOS, iOS 및 watchOS에서 작동합니다. 이것은 NSColor
macOS에서만 작동하고 UIColor
기본적으로 다른 모든 곳에서 작동하는.
Color
SwiftUI NSColor
에서는 UIColor
,, Red / Green / Blue (RGB), Hue / Saturation / Lightness (HSL) 또는 White / Opacity를 사용하여 초기화 할 수 있습니다 .
이러한 이니셜 라이저의 중요한 차이점 Color
은 다른 색상 클래스 에서처럼 "알파"가 아닌 "불투명도"로 투명도 매개 변수 를 레이블링 한다는 것 입니다.
달리 UIColor
, Color
에서 초기화 할 수 없습니다 CGColor
나 CIColor
. 이를 사용하려면 NSColor
(macOS의 경우) 또는 UIColor
(다른 모든 곳)에 대한 이니셜 라이저에 전달하고 결과를 Color
.
다음 예에서는 SwiftUI에 세 가지 유형의 색상을 표시하거나 적어도 SwiftUI가 허용하는 경우 다음과 같이 표시합니다.
import SwiftUI | |
struct ContentView : View { | |
@State var red = CGFloat(0) | |
@State var green = CGFloat(0) | |
@State var blue = CGFloat(0) | |
@State var opacity = CGFloat(1) | |
@State var uiColor = UIColor.clear | |
@State var cgColor = UIColor.clear.cgColor | |
@State var color = Color.clear | |
func setColour() { | |
uiColor = UIColor(red: red, green: green, blue: blue, alpha: opacity) | |
cgColor = CGColor(srgbRed: red, green: green, blue: blue, alpha: opacity) | |
color = Color(red: Double(red), green: Double(green), blue: Double(blue), opacity: Double(opacity)) | |
} | |
var body: some View { | |
VStack { | |
HStack { | |
Text("Red") | |
.frame(width: 100) | |
Slider(value: $red, in: 0...1, step: 0.1, onEditingChanged: { _ in | |
self.setColour() | |
}) | |
} | |
HStack { | |
Text("Green") | |
.frame(width: 100) | |
Slider(value: $green, in: 0...1, step: 0.1, onEditingChanged: { _ in | |
self.setColour() | |
}) | |
} | |
HStack { | |
Text("Blue") | |
.frame(width: 100) | |
Slider(value: $blue, in: 0...1, step: 0.1, onEditingChanged: { _ in | |
self.setColour() | |
}) | |
} | |
HStack { | |
Text("Opacity") | |
.frame(width: 100) | |
Slider(value: $opacity, in: 0...1, step: 0.1, onEditingChanged: { _ in | |
self.setColour() | |
}) | |
} | |
HStack { | |
VStack { | |
Text("UIColor") | |
Rectangle() | |
.foregroundColor(Color(uiColor)) | |
} | |
VStack { | |
Text("CGColor") | |
Rectangle() | |
.foregroundColor(Color(UIColor(cgColor: cgColor))) | |
} | |
VStack { | |
Text("Color") | |
Rectangle() | |
.foregroundColor(color) | |
} | |
} | |
} | |
.padding() | |
} | |
} |
내가 만들 수 있지만 당신이 볼 수 있듯이 CGColor
, UIColor
,와 Color
같은 변수를 사용하여, I가 처음 두를 변환해야합니다 Color
그들은 내에서 사용할 수있는 경우 View
.
또한 Color
이러한 변수는 유형 Double
이 아니라 유형이어야 한다는 것을 알 수 있습니다 CGFloat
.
이것이 Core Graphics Float의 끝의 시작을 나타내는 것처럼 보일 수 있지만 SwiftUI는 여전히 모든 곳에서 사용하며, 아마도 가장 주목할만한 것은 View
프레임 의 너비와 높이를 설정하는 데 사용 됩니다.
아마도 언젠가 SwiftUI는 Double
의 프레임을 설정 하기 위해 를 필요로 할 View
것이며, 그것은 확실히 전환점을 표시 할 것입니다.
ImagePaint
import SwiftUI | |
struct ContentView : View { | |
@State var scale = CGFloat(0.2) | |
let zoomedOut = CGFloat(0.000000001) | |
let zoomedIn = CGFloat(1) | |
var body: some View { | |
VStack { | |
Rectangle() | |
.foregroundColor(.clear) | |
.background(Rectangle().fill(ImagePaint(image: Image("dogs"), scale: scale))) | |
HStack { | |
Button(action: { | |
withAnimation { | |
self.scale = self.zoomedOut | |
} | |
}) { | |
Image(systemName: "minus.magnifyingglass") | |
} | |
Slider(value: $scale, in: 0.000000001...1) | |
Button(action: { | |
withAnimation { | |
self.scale = self.zoomedIn | |
} | |
}) { | |
Image(systemName: "plus.magnifyingglass") | |
} | |
} | |
.padding() | |
} | |
} | |
} |
그라디언트 (선형 / 각도 / 방사형)
import SwiftUI | |
struct ContentView : View { | |
let gradient = Gradient(colors: [.red, .orange, .yellow, .green, .blue]) | |
var body: some View { | |
VStack { | |
ZStack { | |
Rectangle() | |
.foregroundColor(.clear) .background(LinearGradient(gradient: gradient, startPoint: .top, endPoint: .bottom)) | |
TextOverlayView(string: "LinearGradient") | |
} | |
GeometryReader { | |
geometry in | |
ZStack { | |
Rectangle() | |
.foregroundColor(.clear) | |
.background(AngularGradient.init(gradient: self.gradient, center: .zero, angle: Angle(degrees: 180.0))) | |
TextOverlayView(string: "AngularGradient") | |
} | |
} | |
GeometryReader { | |
geometry in | |
ZStack { | |
Rectangle() | |
.foregroundColor(.clear) | |
.background(RadialGradient.init(gradient: self.gradient, center: .zero, startRadius: 45, endRadius: 500)) | |
TextOverlayView(string: "RadialGradient") | |
} | |
} | |
} | |
} | |
} | |
struct TextOverlayView : View { | |
let string: String | |
var body: some View { | |
VStack { | |
Spacer() | |
Text(string) | |
.frame(maxWidth: .infinity, alignment: .leading) | |
.padding() | |
.font(.title) | |
} | |
} | |
} |
GeometryReader 및 GeometryProxy
GeometryReader
View
화면 에서 의 형상을 캡처 할 수 있습니다 .
위의 예 ZStack
에서 전체 View
가 포함 된 자체는 GeometryReader
레이아웃에 영향을주지 않지만 하나의 인수를 전달해야하는 클로저 인에 포함됩니다.
Apple의 튜토리얼 은이 인수를 geometry 라고 부르기 때문에 예제에서도 똑같이했습니다. 전달되는 객체는 GeometryProxy
두 개의 속성과 하나의 메서드를 제공하는입니다.
메서드를 frame(in:)
사용하면 직접 부모 (로컬) 또는 최상위 부모 (글로벌)에 상대적인 프레임 을 전달 CoordinateSpace.local
하거나 CoordinateSpace.global
가져올 수 있습니다.
이 예는 GeometryReader
iPhone 11 Pro Max에서 사용중인 두 개의 s 를 보여줍니다 . 녹색 영역은 VStack
의 첫 번째 부모입니다 View
.
GeometryProxy
로 전달 GeometryReader
제 부모 아래쪽 상단 (44) 및 (34)의 안전 영역 세트를 가지고 고정 프로그램. 상단은 노치 아래에 뷰를 배치하지 않는 것입니다.
홈 버튼이있는 iPhone 8 및 기타 기기에서 상단 안전 영역의 크기는 상태 표시 줄을 피하기 위해 20입니다.
홈 버튼이없는 iPhone에만 34 개의 하단 안전 영역이있어 사용자가 위로 스 와이프하여 집으로 이동하고 앱을 전환 할 수 있습니다.
보기에서 안전 영역을 무시하려면 .edgesIgnoringSafeArea(.all)
수정자를 사용하십시오 . 다른 옵션입니다 .bottom
, .leading
, .top
, 및.trailing.
이러한 옵션에 대한 자세한 내용은 Edge를 참조하십시오 .
대부분의 상황에서 크기가 동일하게 보인다는 점은 주목할 가치가 있습니다. 전역 및 로컬 범위에서 상위 VStack
및 작은 빨간색은 VStack
모두 정적 크기를 갖습니다.
어디 다를 그들의입니다 minX
및 minY
값. 처음에 작은 빨간색을 VStack
안에 넣었을 때 전역 공간 HStack
의 minX
값은 변경되지 않았습니다.
나는 삽입하는 경우에만이 Spacer
한 VStack
상황 글로벌 위치를 모두 만들어 팔에 의해 이동의 권리 minX
와 minY
다르다.
import SwiftUI | |
struct ContentView : View { | |
func displayGeometry(_ geometry: GeometryProxy) -> String { | |
let frameInGlobal = geometry.frame(in: .global) | |
let frameInLocal = geometry.frame(in: .local) | |
return """ | |
safeAreaInsets.top: \t\t\(Int(geometry.safeAreaInsets.top)) | |
safeAreaInsets.leading: \t\(Int(geometry.safeAreaInsets.leading)) | |
safeAreaInsets.bottom: \t\(Int(geometry.safeAreaInsets.bottom)) | |
safeAreaInsets.trailing: \t\(Int(geometry.safeAreaInsets.trailing)) | |
frame(in: .global): (\(String(Int(frameInGlobal.minX))), \(String(Int(frameInGlobal.minY))), \(String(Int(frameInGlobal.size.width))), \(String(Int(frameInGlobal.size.height)))) | |
frame(in: .local): (\(String(Int(frameInLocal.minX))), \(String(Int(frameInLocal.minY))), \(String(Int(frameInLocal.size.width))), \(String(Int(frameInLocal.size.height)))) | |
""" | |
} | |
var body: some View { | |
GeometryReader { | |
geometry in | |
VStack { | |
Spacer() | |
Text("Global geometry") | |
.font(.title) | |
.padding() | |
Text(self.displayGeometry(geometry)) | |
.padding(15) | |
.background(Color.white) | |
.cornerRadius(15) | |
.multilineTextAlignment(.trailing) | |
.foregroundColor(.black) | |
.lineLimit(nil) | |
Spacer() | |
GeometryReader { geometry in | |
VStack { | |
Text("Local geometry") | |
.font(.title) | |
.padding() | |
Text(self.displayGeometry(geometry)) | |
.padding(15) | |
.foregroundColor(.black) | |
.lineLimit(nil) | |
.multilineTextAlignment(.trailing) | |
.background(Color.white) | |
.cornerRadius(15) | |
} | |
.frame(width: 350) | |
.padding() | |
.background(Color.red) | |
.cornerRadius(25) | |
} | |
} | |
.background(Color.green) | |
} | |
} | |
} |
CoordinateSpace
좌표 공간은 위 의 GeometryReader
및 GeometryProxy
섹션 에서 더 자세히 다루었습니다 .
가장 큰 차이점은 a의 로컬 원점 View
이 (0,0)이지만 화면에서의 위치로 인해 View
의 전역 위치가 이것과 다를 수 있다는 것입니다.
CoordinateSpace
객체 의 기능을 사용할 때 .local
및 .global
옵션 을 제공하는 열거 형입니다 .frame(in:)
GeometryProxy
열거 형이 사용되는 유일한 다른 장소는 DragGesture
. 에 대한 이니셜 라이저는 조치를 취하기 전에 필요한 움직임 인 DragGesture
매개 변수 minimumDistance
와 coordinateSpace
열거 형 의 경우를 취할 수 있습니다.
이 coordinateSpace
세트 를 사용하면 제스처 Value
에 대한 onChanged
클로저 로 전달되는 구조체 는 해당 startLocation
및 location
CGPoint
속성 에 상대 또는 범용 좌표를 제공 합니다.
프레임 워크 통합
UIHostingController
Xcode에서 새 SwiftUI 프로젝트를 만들고 SceneDelegate
Swift 파일 로 이동 하면 top 함수가 호출 func scene(_ :, willConnectTo:, options:)
되고 ContentView
구조체 의 인스턴스를 초기화하는 코드가 포함되어 있음을 알 수 있습니다 .
다음으로이 함수 UIWindowScene
는 기본적으로 iPad에서 여러 창 을 사용 하는 관리자 인 current를 가져옵니다 . iPhone이있는 경우 여러 창을 사용할 수 없으므로 하나의 창만 관리하는 것입니다.
if let windowScene = scene as? UIWindowScene
블록 내부에서 SceneDelegate
즉시 현재 창을 가져 오는 것을 알 수 있습니다. 이 시점에서 iPad에 여러 개의 창을 설정하지 않았으므로 창이 하나만 있기 때문에 어렵지 않습니다.
다소 초기처럼 그런 다음 UIViewController
A의 Storyboard
, 우리는 설정 rootViewController
.
새로 생성하면 UIHostingController
SwiftUI를 표시 할 수 있습니다.
같은 자습서가 있습니다 SwiftUI 랩의 삭제 중 조동사 당신에게 당신의 자신의 주문을 생성의 값을 보여줍니다 UIHostingController
.
그보다 더 좋은 예를 제공 할 수는 없지만 그렇게 해결해야 할 특정 문제가 없다면 사용자 지정을 만들 필요가 없습니다 UIHostingController
.
당신이 확인해야합니다 주요 변화는 SceneDelegate
통과하는 EnvironmentObject
당신에 ContentView
. 자세한 내용은을 참조하십시오 EnvironmentObject
.
UIViewRepresentable
UIViewRepresentable
UIKit View
에서 SwiftUI를 만들 수 있습니다 UIViews
.
이 예에서는 TextField
현재 SwiftUI에서 불가능한 여러 줄을 만듭니다. 에 상응하는 것이 없기 때문에 데이터를 영구적으로 저장 하는 간단한 UITextView
방법 UIViewRepresentable
으로 이것을 만듭니다 .ObservableObject
UserDefaults
에 입력하면 MultiTextField
변경 사항이 자동으로 즉시 저장되므로 다음에 앱을 시작할 때 텍스트가 동일합니다.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var text = String() { | |
didSet { | |
UserDefaults.standard.set(text, forKey: "text") | |
} | |
} | |
} | |
struct ContentView : View { | |
var body: some View { | |
MultiTextView() | |
.padding() | |
} | |
} | |
struct MultiTextView: UIViewRepresentable { | |
@ObservedObject var data = DataModel.shared | |
func makeUIView(context: Context) -> UITextView { | |
let view = UITextView() | |
view.isScrollEnabled = true | |
view.isEditable = true | |
view.isUserInteractionEnabled = true | |
view.textAlignment = .left | |
view.delegate = context.coordinator | |
view.font = UIFont.systemFont(ofSize: 18) | |
if let loadedText = UserDefaults.standard.string(forKey: "text") { | |
view.text = loadedText | |
} | |
else { | |
view.text = "Start typing here" | |
} | |
return view | |
} | |
func updateUIView(_ uiView: UITextView, context: Context) { | |
view.text = data.text | |
} | |
func makeCoordinator() -> MultiTextView.Coordinator { | |
Coordinator(self) | |
} | |
class Coordinator: NSObject, UITextViewDelegate { | |
var control: MultiTextView | |
init(_ control: MultiTextView) { | |
self.control = control | |
} | |
func textViewDidChange(_ textView: UITextView) { | |
control.data.text = textView.text | |
} | |
} | |
} |
UIViewControllerRepresentable
UIViewRepresentable
위와 같이 개별 뷰를 나타내는 대신 UIViewController
UIKit에서 전체 인스턴스를 나타낼 수도 있습니다 .
내 예제를 복제하려면 Storyboard
SwiftUI 프로젝트에서을 생성 하고 기본 이름 인 "Storyboard"를 호출해야합니다.
a를 UIViewController
받는 Storyboard
및 뷰 계층 구조를 선택합니다. 설정 Class
에 ViewController
와 Storyboard ID
에 initialVC
.
UILabel
제약 조건을 사용하여 중심에 둔을 추가하십시오 .
이 게시물에서는 UIKit 제약 조건에 대해 설명하지 않겠지 만 온라인에는 많은 자습서가 있습니다.
UILabel
"이것은 Storyboard의 ViewController입니다"라고 맨 위에 추가 한 것이 필요하지 않습니다 . 이것은 작동 할 때 명확하게하기위한 것입니다.
또한 배경을 파란색으로 만들 필요가 없습니다 UIViewController
. SwiftUI와 .
import SwiftUI | |
struct ContentView : View { | |
@State var text = "" | |
var body: some View { | |
VStack { | |
TextField("Add text here", text: $text) | |
.padding() | |
UIKitVC(text: $text) | |
} | |
} | |
} | |
struct UIKitVC: UIViewControllerRepresentable { | |
@Binding var text: String | |
func makeUIViewController(context: Context) -> ViewController { | |
let storyboard = UIStoryboard(name: "Storyboard", bundle: nil) | |
let viewController = storyboard.instantiateViewController(withIdentifier: "initialVC") as! ViewController | |
return viewController | |
} | |
func updateUIViewController(_ uiViewController: ViewController, context: Context) { | |
uiViewController.labelText = text | |
} | |
} |
ViewController
에서 인스턴스 를 만들 필요는 없습니다 Storyboard
. 저는 이것이 많은 사람들에게 유용한 예라고 생각했습니다.
당신은에 포함 할 모든 것을 UIViewConrollerRepresentable
IS makeUIViewController(context:)
초기화되는 ViewController
및 updateUIViewController(_:, context:)
SwiftUI의 변화에 어떤 반응하여합니다.
공지 사항 내가 가지고 @Binding
내 연결되어 재산 ContentView
나는 SwiftUI에 만들어 그 변경을 의미 구조체 TextField
에서이 속성을 업데이트를 UIKitVC
하면서, updateUIViewController(_:, context:)
실제 이러한 변경 통과 UILabel
에서 ViewController
클래스를.
DigitalCrownRotationalSensitivity
이것은 Apple Watch 측면에있는 회전 다이얼의 감도 목록입니다. 그것은 오면 .low
, .medium
및 .high
품종.
상태 및 데이터 흐름
상태
View
구조체에 로컬로 저장하려는 모든 변수 는 이것으로 표시되어야합니다. 변수를 추가하고 추가하지 않으면 @State
컨트롤 값을 저장하는 데 사용할 수 없습니다.
이는 @State
변수가 런타임에 변경 될 수 있고 SwiftUI가이 View
를 기반으로 다시 그려 지기 때문 입니다.
예를 들어, 당신은이 경우 Button
A와 바인딩 값 변경이 Slider
, Slider
당신이에게 이동하지 못했다는 사실에도 불구하고 이루어진 변경 반영하기 위해 이동 것 Slider
자체를.
다음은 해당 @State
변수 가있는 세 가지 주요 SwiftUI 컨트롤과 Button
모두 변경 하는 재설정 입니다.
import SwiftUI | |
struct ContentView : View { | |
@State var toggleIsOn = false | |
@State var sliderValue = 0.0 | |
@State var stepperValue = 0 | |
var body : some View { | |
VStack { | |
Text("Slider: \(sliderValue)") | |
Slider(value: $sliderValue, in: 0...100, step: 0.1) | |
Stepper(value: $stepperValue, in: 0...100, step: 1) { | |
Text("Stepper: \(stepperValue)") | |
} | |
Toggle(isOn: $toggleIsOn) { | |
Text("Toggle: \(String(toggleIsOn))") | |
} | |
} | |
.padding() | |
} | |
} |
제본
@State
자식의 부모에 있는 속성에 영향을 미치려면 해당 속성 View
을 전달하고로 표시해야합니다 @Binding
. 이것은 부모가 가진 것과 같은 View
부모의 @State
변수에 대한 동일한 직접 액세스를 자식에게 제공합니다 .
아래 예에서는 위의 예를 사용하여 @State
child를 포함하는 시트를 제공합니다 View
.
로컬 @Binding
속성이 초기화되지 않았으므로 PresentedView
시트 수정 자에서 구조체 를 만들 때 초기화해야합니다 .
import SwiftUI | |
struct ContentView : View { | |
@State var toggleIsOn = false | |
@State var sliderValue = 0.0 | |
@State var stepperValue = 0 | |
@State var presented = false | |
var body : some View { | |
VStack { | |
Text("Slider: \(sliderValue)") | |
Slider(value: $sliderValue, in: 0...100, step: 0.1) | |
Stepper(value: $stepperValue, in: 0...100, step: 1) { | |
Text("Stepper: \(stepperValue)") | |
} | |
Toggle(isOn: $toggleIsOn) { | |
Text("Toggle: \(String(toggleIsOn))") | |
} | |
Button(action: {self.presented = true}) { | |
Text("Present") | |
} | |
} | |
.sheet(isPresented: $presented, content: {PresentedView(sliderValueFromParent: self.$sliderValue, stepperValueFromParent: self.$stepperValue, toggleIsOnFromParent: self.$toggleIsOn)}) | |
.padding() | |
} | |
} | |
struct PresentedView : View { | |
@Binding var sliderValueFromParent: Double | |
@Binding var stepperValueFromParent: Int | |
@Binding var toggleIsOnFromParent: Bool | |
@Environment(\.presentationMode) var presentationMode | |
var body: some View { | |
VStack { | |
Text("Slider: \(sliderValueFromParent)") | |
Slider(value: $sliderValueFromParent, in: 0...100, step: 0.1) | |
Stepper(value: $stepperValueFromParent, in: 0...100, step: 1) { | |
Text("Stepper: \(stepperValueFromParent)") | |
} | |
Toggle(isOn: $toggleIsOnFromParent) { | |
Text("Toggle: \(String(toggleIsOnFromParent))") | |
} | |
Button(action: {self.presentationMode.wrappedValue.dismiss()}) { | |
Text("Dismiss") | |
} | |
} | |
} | |
} |
다음 줄에 대해 잘 모르는 경우 :
Environment(\.presentationMode) var presentationMode
ObservedObject
같은 컨트롤에 대한 나의 예에서 Toggles
와 TextFields
, 나는 준수하는 스위프트 클래스의 데이터에 액세스 할 수있는 간단한 방법 보여 주었다 ObservableObject
프로토콜을.
단순성을 위해 동일한 코드 스 니펫으로 보여 주지만 실제로는 별도의 Swift 파일에 있어야합니다. 다음은 Swift 클래스를 @ObservedObject
.
즉, @Published
Swift 파일에 표시된 변수를 변경 하면 그에 따라 뷰를 업데이트하도록 SwiftUI 파일에 알립니다.
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var myObservedString = "Hello world!" | |
} | |
struct ContentView: View { | |
@ObservedObject var data = DataModel.shared | |
var body: some View { | |
Text(data.myObservedString) | |
} | |
} |
EnvironmentObject
를 추가하는 EnvironmentObject
것은 ObservedObject
.
의 구조 DataModel
여기 클래스는 정확히 동일하지만, 우리는로 표시되어 EnvironmentObject
내부 ContentView
및 하지 로 설정 DataModel.shared
.
대신, 우리는 이름과 유형으로 선언 DataModel.shared
하고 SceneDelegate
Swift 파일 을 사용하여 전달 됩니다.
SceneDelegate
전달 하기 위해 변경해야 할 사항을 포함 시켰 EnvironmentObject
습니다. NavigationLink
시트를 사용하거나 표시하는 것과 같이 탐색하는 모든 후속 뷰 EnvironmentObject
는 생성 될 때 동일한 뷰가 전달되어야합니다.
버튼을 눌러 표시 할 수있는 시트를 예제에 추가했습니다.이 시트는 어떻게 environmentObject
전달 되는지 보여줍니다 (에서와 동일하지만 SceneDelegate
).
/* | |
Go to SceneDelegate.swift and change the top function to this: | |
func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) { | |
let contentView = ContentView() | |
if let windowScene = scene as? UIWindowScene { | |
let window = UIWindow(windowScene: windowScene) | |
//This is the only line that you change | |
window.rootViewController = UIHostingController(rootView: contentView.environmentObject(DataModel.shared)) | |
self.window = window | |
window.makeKeyAndVisible() | |
} | |
} | |
*/ | |
import SwiftUI | |
final class DataModel: ObservableObject { | |
static let shared = DataModel() | |
@Published var myObservedString = "Hello world!" | |
} | |
struct ContentView: View { | |
@EnvironmentObject var data: DataModel | |
@State var presented = false | |
var body: some View { | |
VStack { | |
Text(data.myObservedString) | |
Button(action: {self.presented.toggle()}) { | |
Text("Present") | |
} | |
} | |
.sheet(isPresented: $presented, content: { | |
PresentedView().environmentObject(self.data) | |
}) | |
} | |
} | |
struct PresentedView : View { | |
@EnvironmentObject var data: DataModel | |
@Environment(\.presentationMode) var presentationMode | |
var body : some View { | |
VStack { | |
Text(data.myObservedString) | |
Button(action: | |
{self.presentationMode.wrappedValue.dismiss()} | |
) { | |
Text("Dismiss") | |
} | |
} | |
} | |
} |
FetchRequest 및 FetchedResults
에 대한 좋은 Hacking With Swift 튜토리얼이 있습니다FetchRequest
.
DynamicProperty
View
기본 데이터가 변경 될 때 SwiftUI 를 새로 고치려면이 기본 개념을 캡슐화하는 프로토콜이 필요합니다.
순응 유형 중 적어도 일부는 Binding
, Environment
, EnvironmentObject
, FetchRequest
, GestureState
, ObservedObject
, 및 State
SwiftUI의 데이터를 사용했다 누구나 잘 알고 있어야합니다.
이러한 속성은의 본문 View
이 다시 그려 지기 전에 설정 됩니다.
환경
환경을 사용하면 기기 설정과 관련된 설정에 액세스 할 수 있습니다. 예를 들어, ColourScheme.dark
새로운 다크 모드에서 앱이 어떻게 보일지 미리보고 기존 .NET과 나란히 대조 할 수 있습니다 ColourScheme.light
.
접근성 설정에서 대비를 높일 수 있으므로 ColorSchemeContrast.increased
또는 ColorSchemeContrast.standard
유일한 옵션입니다.
접근성 설정에서 굵은 텍스트를 선택한 결과는로 표시되며 LegibilityWeight.bold
그렇지 않은 경우 기본값은 LegibilityWeight.regular
입니다.
Every View
에는 PresentationMode
하나의 속성을 저장하고 하나의 메서드 가있는 연결된 구조체가 있습니다. 속성은 bool이며이 활성화되어 isPresented
있는지 여부를 알려줍니다 View
.
방법은이며 dismiss()
, View
화면 에서 전류를 제거하고 View
표시 한 것으로 돌아갈 수 있습니다.
계속할 수는 있지만 모든 단일 Environment
값 을 바인딩 하고 데이터를 보는 방법을 나열하는 예제를 제공하는 것이 더 합리적 일 것입니다 List
.
import SwiftUI | |
struct ContentView: View { | |
@Environment(\.colorScheme) var colorScheme | |
//.dark or .light | |
@Environment(\.colorSchemeContrast) var colorSchemeContrast | |
//.increased or .standard | |
@Environment(\.legibilityWeight) var legibilityWeight | |
//.bold or .standard | |
@Environment(\.presentationMode) var presentationMode | |
//.isPresented = true or false | |
//.dismiss() ends presentation of View | |
@Environment(\.editMode) var editMode | |
//.active when content can be edited | |
//.inactive when content cannot be edited | |
//.transient | |
@Environment(\.horizontalSizeClass) var horizontalSizeClass | |
//.compact or .regular | |
@Environment(\.verticalSizeClass) var verticalSizeClass | |
//.compact or .regular | |
@Environment(\.disableAutocorrection) var disableAutocorrection | |
//If autocorrection is enabled (false) or disabled (true) | |
@Environment(\.sizeCategory) var sizeCategory | |
//Font size | |
//.extraSmall, .small, .medium, .large, .extraLarge, .extraExtraLarge, .extraExtraExtraLarge, .accessibilityMedium, .accessibilityLarge, .accessibilityExtraLarge, .accessibilityExtraExtraLarge, .accessibilityExtraExtraExtraLarge | |
@Environment(\.managedObjectContext) var managedObjectContext | |
//Management for persistent data storage | |
@Environment(\.undoManager) var undoManager | |
//nil if undo/redo disabled, otherwise manages undo/redo | |
@Environment(\.layoutDirection) var layoutDirection | |
//.leftToRight or .rightToLeft | |
@Environment(\.defaultMinListRowHeight) var defaultMinListRowHeight | |
//Minimum height of a List row (system default if nil) | |
@Environment(\.defaultMinListHeaderHeight) var defaultMinListHeaderHeight | |
//Minimum height of a List header (system default if nil) | |
@Environment(\.isEnabled) var isEnabled | |
//Whether user interaction is enabled (true) or disabled (false) | |
@Environment(\.font) var font | |
//The font for the View | |
@Environment(\.displayScale) var displayScale | |
//Amount that text size is increased to, maximum is 2 | |
@Environment(\.pixelLength) var pixelLength | |
//Equal to 1 divided by display scale | |
@Environment(\.locale) var locale | |
//.isRegionCodes, isoLanguageCodes, isoCurrencyCodes | |
@Environment(\.calendar) var calendar | |
//The current calendar type | |
//.buddhist, .chinese, .coptic, .ethiopicAmeteAlem, .ethiopicAmeteMihret, .gregorian, .hebrew, .indian, .islamic, .islamicCivil, .islamicTabular, .islamicUmmAlQura, .iso8601, .japanese, .persian, .republicOfChina | |
//The type of calenar | |
@Environment(\.timeZone) var timeZone | |
//The current time zone | |
@Environment(\.accessibilityEnabled) var accessibilityEnabled | |
//Whether one or more accessibility features is enabled (true) or disabled (false) | |
@Environment(\.accessibilityDifferentiateWithoutColor) var accessibilityDifferentiateWithoutColor | |
//Whether differentiate colour is enabled (true) or disabled (false) in Accessibility settings | |
@Environment(\.accessibilityReduceTransparency) var accessibilityReduceTransparency | |
//Whether reduce transparency is enabled (true) or disabled (false) in Accessibility settings | |
@Environment(\.accessibilityReduceMotion) var accessibilityReduceMotion | |
//Whether reduce motion is enabled (true) or disabled (false) in Accessibility settings | |
@Environment(\.accessibilityInvertColors) var accessibilityInvertColors | |
//Whether invert colours is enabled (true) or disabled (false) in Accessibility settings | |
@Environment(\.multilineTextAlignment) var multilineTextAlignment | |
//How Text is aligned when it has an explicit width | |
//.center, .leading or .trailing | |
@Environment(\.truncationMode) var truncationMode | |
//Which part of the text that becomes '...' when out of space | |
//.head, .middle or .tail | |
@Environment(\.lineSpacing) var lineSpacing | |
//Spacing between lines of a Text | |
@Environment(\.allowsTightening) var allowsTightening | |
//Whether letter spacing will be tightened on a Text | |
@Environment(\.lineLimit) var lineLimit | |
//The maximum number of lines of a Text, default is nil (no limit) | |
@Environment(\.minimumScaleFactor) var minimumScaleFactor | |
//The smallest a View can be scaled between 0 and 1 | |
//MacOS only | |
//@Environment(\.controlActiveState) var controlActiveState | |
@State var keys: [Any] = [""] | |
@State var presented = false | |
let names = ["colorScheme", "colorSchemeContrast", "legibilityWeight", "presentationMode", "editMode", "horizontalSizeClass", "verticalSizeClass", "disableAutocorrection", "sizeCategory", "managedObjectContext", "undoManager", "layoutDirection", "defaultMinListRowHeight", "defaultMinListHeaderHeight", "isEnabled", "font", "displayScale", "pixelLength", "locale", "calendar", "timeZone", "accessibilityEnabled", "accessibilityDifferentiateWithoutColor", "accessibilityReduceTransparency", "accessibilityReduceMotion", "accessibilityInvertColors", "multilineTextAlignment", "truncationMode", "lineSpacing", "allowsTightening", "lineLimit", "minimumScaleFactor"] | |
func setArray() { | |
if keys.count == 1 { | |
self.keys = [self.colorScheme, self.colorSchemeContrast, self.legibilityWeight ?? .regular, self.presentationMode, self.editMode?.wrappedValue ?? EditMode.inactive, self.horizontalSizeClass ?? .regular, self.verticalSizeClass ?? .regular, self.disableAutocorrection ?? false, self.sizeCategory, self.managedObjectContext, self.undoManager ?? UndoManager(), self.layoutDirection, self.defaultMinListRowHeight, self.defaultMinListHeaderHeight ?? 0, self.isEnabled, self.font ?? Font.body, self.displayScale, self.pixelLength, self.locale, self.calendar, self.timeZone, self.accessibilityEnabled, self.accessibilityDifferentiateWithoutColor, self.accessibilityReduceTransparency, self.accessibilityReduceMotion, self.accessibilityInvertColors, self.multilineTextAlignment, self.truncationMode, self.lineSpacing, self.allowsTightening, self.lineLimit as Any, self.minimumScaleFactor] | |
} | |
} | |
var body: some View { | |
VStack { | |
HStack { | |
EditButton() | |
Button(action: {self.presented.toggle()}) { | |
Text("Present") | |
} | |
} | |
List { | |
ForEach(0..<keys.count, id: \.self) { index in | |
VStack(alignment: .leading) { | |
Text("\(self.names[index])") | |
.font(.title) | |
Text("\(String(describing: self.keys[index]))") | |
} | |
.onAppear(perform: {self.setArray()}) | |
} | |
} | |
} | |
.sheet(isPresented: $presented, content: {PresentedView()}) | |
} | |
} | |
struct PresentedView : View { | |
@Environment(\.presentationMode) var presentationMode | |
//.isPresented = true or false | |
//.dismiss() ends presentation of View | |
var body : some View { | |
VStack { | |
Text("presentationMode") | |
.font(.title) | |
Text(String(describing: presentationMode)) | |
Button(action: { | |
self.presentationMode.wrappedValue.dismiss() | |
}) { | |
Text("Dismiss") | |
} | |
} | |
} | |
} |
PreferenceKey
누구나 View
수정자를 사용하여 .preferredColorScheme(.dark)
장치가 라이트 모드로 설정되어 있어도 다크 모드의 모양이되도록 강제 할 수 있습니다 .
이것이하는 일은 PreferredColorSchemeKey
구조체의 value 속성을의 속성으로 설정하는 것입니다 ColorScheme.dark
. 에서 View
라이트 모드를 사용 하도록하여 재정의 할 수 있습니다. .colorScheme(.light)
이것이 첫 번째 수정자가 필수 상태가 아닌 기본 설정 만 나타내는 이유입니다.
.preferredColorScheme(nil)
기본 색 구성표가 사용되는 반면 .colorScheme(nil)
호출 할 수없는 기본 색 구성표가없는 기본 설정을 나타 내기 위해 호출 할 수 있습니다.
왜 전화를 걸 수 .colorScheme(nil)
없습니까?
PreferredColorSchemeKey
PreferenceKey
값뿐만 아니라 defaultValue
값이 설정되지 않은 경우에도 사용할 수있는 프로토콜을 준수합니다 .
.colorScheme(.light)
수정은 전혀 구조체 값을 설정하고 단지를 반환하지 않습니다 View
필요한 색상으로.
LocalizedStringKey
LocalizedStringKey
문자열에서 만들 수 있으며이를 Localisable.strings
사용하여 국제화에 사용되는 다른 파일이나 해당 값을 찾으려고합니다 .
값이 없으면 키 자체가 대신 사용됩니다.
사용 예는 텍스트를LocalizedStringKey
참조하십시오 .
제스처
제스처
나는 이것이 그들의 문서를 대체해야한다는 것을 알고 있지만 Apple의 튜토리얼 은 Gestures를 꽤 잘 다루고 있습니다. 제공된 것과 유사한 예제를 제공하고 싶지 않으므로이 섹션을 그대로 두겠습니다.
기본 사항을 마스터하면 Apple은 제스처를 더 복잡한 상호 작용으로 결합하는 또 다른 튜토리얼 을 제공합니다. 이 두 기사 모두 하단에 표준 제스처 유형 목록이 있으며, 모두 매우 잘 문서화되어있는 것 같습니다.
추가 설명이 필요한 제스처 문서 페이지를 찾으면 알려 주시면 최선을 다해 여기에서 설명하겠습니다.
미리보기
PreviewProvider 프로토콜
이 공급자를 준수하는 구조체를 만들면의 컬렉션을 만들 수 있습니다 View
. 를 만들면 Group
각각 다른 플랫폼을 가질 수있는 여러 미리보기를 만들 수 있습니다.
당신은 사용할 수 있습니다 VStack
, HStack
그리고 ZStack
이것에 대한,하지만 당신은 다른 미리보기 장치를 선택하는 경우에도 하나의 화면에 여러 개의 미리보기를 표시하는 기괴한 결과를 생성합니다.
struct ContentView_Previews: PreviewProvider { | |
static var previews: some View { | |
Group { | |
ContentView() | |
.previewDevice("iPhone X") | |
ContentView() | |
.previewLayout(.fixed(width: 1000, height: 600)) | |
ContentView() | |
.previewLayout(.sizeThatFits) | |
ContentView() | |
.environment(\.colorScheme, .dark) | |
ContentView() | |
.environment(\.colorScheme, .light) | |
ContentView() | |
.environment(\.locale, .init(identifier: "en")) | |
} | |
} | |
} |
장치를 지정할뿐만 아니라 플랫폼을 지정할 수도 있습니다. 이에 대한 현재 옵션은 iOS, macOS, tvOS 및 watchOS입니다.
기본적으로 PreviewLayout
값은로 설정되어 .device
장치의 모양을 표시하고 그 안에 미리보기에 맞 춥니 다. 로 설정하면 .sizeThatFits
컨테이너에 디바이스 크기가 제공되는 것처럼 보이지만 컨테이너 주변에 디바이스 베젤이 표시되지 않습니다.
마지막으로 설정 previewLayout
하는 것은 fixed
당신이 당신도 당신의 일에 대해 방해하지 않을 때 유용 할 수있는 컨테이너에 대한 사용자 정의 너비와 높이를 설정할 수 있습니다 View
장치에 같이됩니다.
를 사용 .environment
하면 다크 모드에서 미리보고 앱이 다른 현지화에서 어떻게 작동하는지 확인할 수 있습니다.
읽어 주셔서 감사합니다!
Suggested posts
API 작업에 유용한 Python 코드 조각 3 개.
코드 효율성과 견고성을 개선하십시오.
마이크로 서비스의 출현으로 오케스트레이션 된 시스템이든 데이터 추출이든 상호 작용을위한 많은 API가 있습니다. 전용 APIS 워크로드에 Python을 사용하는 것은 다재다능한 특성 때문에 적절합니다.
자바의 레코드와 Kotlin의 데이터 클래스
자바의 레코드 클래스가 Kotlin의 데이터 클래스와 비교되는 방식입니다. 최근 Java 릴리스에는 레코드 클래스 개념을 포함하여 매우 주목할만한 개발자 친화적 인 기능이 있습니다.
Related posts
세그먼트 컨트롤 및 SwiftUI 2.0
세그먼트 화 된 컨트롤의 값이 변경 될 때 UI 애니메이션
전체보기를 한 세그먼트에서 다른 세그먼트로 스크롤 할 수있는 완전히 분할 된 컨트롤보기는 없습니다. 그러나 SwiftUI로 거의 모든 것을 달성 할 수 있기 때문에 그것은 완전히 괜찮습니다.
내 iOS 앱 생성 프로세스 — 4 부
소개 먼저 SwiftUI를 좋아한다고 말씀 드리겠습니다! 내가 SwiftUI의 팬인 이유는 무엇입니까? 명령 적 (인터페이스 빌더) 접근 방식보다 인터페이스 개발에 대한 선언적 접근 방식을 선호합니다. SwiftUI를 사용하면 iOS 시뮬레이터를 실행하지 않고도 디자인 아이디어를 더 빠르게 반복하고 결과를 볼 수 있습니다.
Enum을 효과적으로 사용
열거 형을 적절히 사용하여 데이터를 모델링하는 방법
Swift는 흥미로운 언어입니다. 새로운 개발자에게 매우 친숙합니다. 구문은 간결하고 이해하기 쉬우 며 주요 개념도 있습니다.
SwiftUI에서 Segue에 위임 사용
보기를 변경하는 새로운 방법
나는 SwiftUI에서보기 변경에 대한 기사를 썼습니다. 그 안에서 선호하는 방법은 Published ObservableObjects를 사용하는 것이 었습니다.
Swift에서 작업 종속성은 무엇입니까?
앱의 비즈니스 로직 분리 및 모듈화
이 기사에서는 작업 종속성과 이러한 종속성이 여러 작업을 연결하여 분리되고 재사용 가능한 비즈니스 논리 구성 요소를 만드는 데 어떻게 도움이되는지 알아 봅니다. 이미지를로드하는 작업과 세피아 효과를 추가하는 작업의 두 가지 작업을 연결합니다.
댓글 없음:
댓글 쓰기