I’ll continue the SwiftUI tutorial with some interpretation. The original | address
Tutorial 2 – Building Lists and Navigation
Section 4-step 2: StaticList
var body: some View {
List {
LandmarkRow(landmark: landmarkData[0])
LandmarkRow(landmark: landmarkData[1])
}
}
Copy the code
The List here is similar to a container like HStack or VStack in that it takes a View Builder and lists two LandmarkRows using a View DSL. This approach builds the organization of static cells corresponding to UITableView.
public init(content: () -> Content)
Copy the code
We can run the app and use Xcode’s View Hierarchy tool to View the UI, and the results may look familiar:
Actually the UpdateCoalesingTableView that’s drawn on the screen is a subclass of UITableView, and the two cell ListCoreCellHost are subclasses of UITableViewCell. For lists, SwiftUI uses a set of implementation logic directly from a mature UITableView, rather than redrawing. Like a Text or Image, by contrast, the single View in the UIKit layer are all unified by the DisplayList. ViewUpdater. Platform. CGDrawingView to draw the UIView subclass.
However, the first thing we need to do when using SwiftUI is to think outside of UIKit and not worry about the drawing and implementation behind it. Using a UITableView to express a List may be a stopgap, and may be replaced in the future by a more efficient way of drawing. Because the SwiftUI layer is just an abstraction of the data described by the View, like the React Virtual DOM and Flutter widgets, the drawing behind the SwiftUI layer is completely decoupled and interchangeable. This leaves plenty of possibilities for SwiftUI in the future.
Section 5-Step 2: DynamicList
和 Identifiable
List(landmarkData.identified(by: .id)) { landmark in
LandmarkRow(landmark: landmark)
}
Copy the code
In addition to static input, the List can also accept dynamic input, which uses a different initialization method than static:
public struct List<Selection, Content> where Selection : SelectionManager, Content : View {
public init<Data, RowContent>(
_ data: Data, action: @escaping (Data.Element.IdentifiedValue) -> Void,
rowContent: @escaping (Data.Element.IdentifiedValue) -> RowContent)
where
Content == ForEach<Data, Button<HStack<RowContent>>>,
Data : RandomAccessCollection,
RowContent : View,
Data.Element : Identifiable
//...
}
Copy the code
This initialization method has more constraints, so let’s look at it line by line:
Content == ForEach<Data, Button<HStack<RowContent>>>
Because it’s not in the function signatureContent
.Content
onlyList<Selection, Content>
Is defined in the type declaration, so this is not so much a constraint as a reverse determinationList
Description of the actual type. Let’s focus on the more important things for now, and we’ll talk more about this later.Data : RandomAccessCollection
This is basically the same thing as asking for the first input parameter to beArray
.RowContent : View
For building each rowrowContent
Say, need to return isView
It’s a normal thing. Pay attention torowContent
In fact, it was also@ViewBuilder
Marked, so you can also putLandmarkRow
Expand and write in. However, we generally prefer to dismantle small UI widgets as much as possible, rather than pile things together.Data.Element : Identifiable
requirementsData.Element
(that is, the type of an array element) exists on which an instance can be identifiedmeetHashable
The id of the. This requirement enables you to quickly locate the cell corresponding to the changed data and refresh the UI when data changes.
There’s an interesting fact about lists and other common base Views. In the following code, we expect the List initializer to generate a View of some type:
var body: some View { List { //... }}Copy the code
But if you look through the List documentation, or even Cmd + Click to SwiftUI’s Interface for a View, you won’t find a declaration like List: View.
Is it because SwiftUI did something to “act” as a View that doesn’t already fit the View type? Of course not… If you use LLDB to print the List type information at runtime, you can see the following information:
(lldb) type lookup List
...
struct List<Selection, Content> : SwiftUI._UnaryView where ...
Copy the code
Further, _UnaryView declares:
protocol _UnaryView : View where Self.Body : _UnaryView {
}
Copy the code
The unary View protocol within SwiftUI is hidden from View, while the List that satisfies it is public, but the protocol chain can be hidden from View as well. This is a privilege of Swift’s internal framework, and third-party developers cannot thus insert a private declaration between two public declarations.
Finally, SwiftUI currently (Xcode 11 Beta 1) only has lists that correspond to uITableViews, not uicollectionViews that correspond to types like Grid. To achieve a similar effect today, you can only nest VStack and HStack. This is strange, because technically it should not be much different from Table View, probably because the time limit is not enough? Grid should be added in the future.
Tutorial 3 – Handling User Input
Section 3 – Step 2: @State
和 Binding
@State var showFavoritesOnly = true var body: some View { NavigationView { List { Toggle(isOn: $showFavoritesOnly) { Text("Favorites only") } //... if ! self.showFavoritesOnly || landmark.isFavorite {Copy the code
Two features appear that were not previously available in Swift: @state and $showFavoritesOnly.
If you Click Cmd into the definition of State, you can see that it is actually a special struct:
@propertyWrapper public struct State<Value> : DynamicViewProperty, BindingConvertible {
/// Initialize with the provided initial value.
public init(initialValue value: Value)
/// The current state value.
public var value: Value { get nonmutating set }
/// Returns a binding referencing the state value.
public var binding: Binding<Value> { get }
/// Produces the binding referencing this state value
public var delegateValue: Binding<Value> { get }
}
Copy the code
The @propertyWrapper annotation is similar to the @_FunctionBuilder mentioned in the previous article in that the struct it decorates can become a new decorator and be applied to other code to change its default behavior. Here the @propertyWrapper State modifier is used as the @state modifier and is used to modify the showFavoritesOnly variable in the View.
Unlike @_FunctionBuilder, which is responsible for “reconstructing” functions according to the rules, the @propertyWrapper decorator is ultimately applied to properties to “wrap” them and control the read and write behavior of a property. If you “expand” the code, it actually looks like this:
// @State var showFavoritesOnly = true var showFavoritesOnly = State(initialValue: true) var body: some View { NavigationView { List { // Toggle(isOn: $showFavoritesOnly) { Toggle(isOn: showFavoritesOnly.binding) { Text("Favorites only") } //... // if ! self.showFavoritesOnly || landmark.isFavorite { if ! self.showFavoritesOnly.value || landmark.isFavorite {Copy the code
I commented out the section before the change and wrote the expanded result on the next line. You can see that @State is just a shorthand way of declaring a State struct. The rules for how to read and write attributes are defined in State. For reading, it is very simple: use showFavoritesOnly. Value to get the actual value stored in State. And the original code written $showFavoritesOnly is showFavoritesOnly. Binding of simplified. Binding will create a reference to showFavoritesOnly and pass it to Toggle. Again, this binding is a reference type, so modifying it in Toggle directly reflects the showFavoritesOnly value of the current View. State’s value didSet triggers the body refresh to complete the State -> View binding.
In Xcode 11 Beta 1, the name of the decorator used in Swift was @propertyDelegate, but in WWDC Apple referred to this feature as @propertyWrapper. According to reliable sources, it will also be called @propertyWrapper in future releases, so a simple mapping should also be suggested when looking at various materials.
If you want more details about @propertyWrapper, check out the related proposals and forum discussions. The interesting detail is that Apple put the proposal back into the “revised” status after the corresponding PR merge into the Master, rather than accepting it directly. There should be some other details besides the @propertyWrapper name fix, but there shouldn’t be much change in the behavior patterns already exposed.
There are also several common @ modifier in SwiftUI, such as @binding, @environment, @environmentobject, etc. The principle of SwiftUI is the same as @state, but their corresponding structs define the read and write mode differently. Together, they form the most basic unit of SwiftUI data flow. For SwiftUI’s data flow, it would cover an entire article if expanded. Session 226-Data Flow Through SwiftUI is highly recommended.
Tutorial 7 – Working with UI Controls
Section 4-Step 2: AboutView
Life cycle of
ProfileEditor(profile: $draftProfile)
.onDisappear {
self.draftProfile = self.profile
}
Copy the code
When WE’re developing UIKit, we’re often exposed to life cycle methods like viewDidLoad, viewWillAppear, and do configuration in them. SwiftUI also has some of these lifecycle methods, such as.onAppear and.onDisappear, which are “unified” under the modifier umbrella.
SwiftUI, however, has fewer and more generic hook lifecycle methods than UIKit. Doing things within the lifecycle itself goes against the declarative concept of programming and looks like adding imperative hacks. Personally, I would like to see View and Combine a little more deeply to bind life-cycle operations like self.draftProfile = self.profile.
A more general event response hook than.onAppear and.onDisappear is.onreceive (_: Perform :), which defines an arbitrary View that responds to the target Publisher, OnReceive is called as soon as the subscribed Publisher issues a new event. Since we can define these publishers ourselves, it is complete, which is useful when converting existing UIKit Views to SwiftUI Views.