UITableView is like bread and butter for (almost) all iOS developers. In most cases, we use a UITableViewCell to display a data type and reuse cells through Identifiers. This technique is introduced in objc.io. When we want to use multiple cells of different types in a Table View, the situation is much more complicated. The inconsistency of cells makes it difficult for us to handle. This article introduces three ways to solve this problem. Each solution attempts to fix the problems caused by the previous solution. The first method is common in many Objective-C code bases. The second method uses enumerations, but it is still not a perfect solution. The third method is implemented using protocols and generics - they are the magic weapons provided to us by Swift. Base I will walk you through a demo project (github address). In this example, we create a Table View with two different cells: one cell for displaying text and one cell for displaying images, as shown below: UITableView showing two types of data (text and pictures) When rendering a view, I like to use value types to encapsulate data. I call this view data. Here, we use two view data:
(In a real project, there may be more properties; the image property should be declared as NSURL to avoid dependence on UIKit.) Correspondingly, we also need two types of cells to display these two types of view data:
Then, we start to enter the View Controller. ***Method: Simple method I don't like to talk about complicated things at the beginning. At the beginning, I will talk about a simple implementation to display something on the screen. We want the Table View to be driven by the data in the array (the items array, to be precise). Because our data is two completely different structures, the array type can only be [Any]. In the registerCells() method, we use the standard cell reuse mechanism to register the cell in advance. In the tableView(_:cellForRowAtIndexPath:) method, we create a cell based on the type of view data corresponding to the specified IndexPath. The complete implementation of our View Controller is very simple (for simplicity, we use ViewController as the data source of the Table View. In a real project, we may need to extract the data source into a separate object.):
This approach works, but I am not happy with it for at least the following reasons:
1. Add a new reusable Identifier 2. Modify the registerCells() method 3. Modify the tableView(\_:cellForRowAtIndexPath:) method
The second method: enumeration We can add a TableViewItem enumeration type to solve these problems to some extent. In the enumeration, we list all the types supported by view data:
Then change the type of the items property to [TableViewItem]:
Then modify the registerCells() method:
***, modify the tableView(_:cellForRowAtIndexPath:) method:
Admittedly, this approach is better than the previous one:
Then we can also improve this implementation, such as modifying the reuse and settings of cells to:
But the sad thing is, we have to add these switch statements everywhere. So far, we have only used switch statements in two places, but it is not difficult to imagine that it is more than just that. For example, when automatic layout will become unavailable and we have to use manual layout, we must use another switch statement in tableView(\_:heightForRowAtIndexPath:). It's not that this method can't be used, but I'm still bothered by those switch statements, so I decided to take it a step further. The third (***) approach: protocols and generics Let’s completely overturn the first two solutions and start over. Declaring the Updatable Protocol Our cell presents different interfaces based on view data, so we define an Updatable protocol and bind it to a type ViewData:
Then let our custom cell implement this protocol:
After looking at the first two methods, it is not difficult to find that for each view data object in items, we need: 1. Find out which cell class to use 2. Find out which reuse identifier to use 3. Rendering cells with view data Define the CellConfigurator structure Therefore, we declare another structure to package the view data. Use the structure to provide more properties and functions. Let's name this structure CellConfigurator:
This is a generic structure that uses the type parameter Cell. Cell has two constraints: first, it must implement the Updatable protocol, and second, it must be a subclass of UITableViewCell. CellConfigurator has three properties: viewData, reuseIdentifier and cellClass. The type of viewData depends on the type of Cell and it has no default value. The values of the other two properties depend on the specific type of Cell (this is a new feature in Swift, it's really great!).
***, we pass the UITableViewCell instance to the updateCell() method, and we can render the viewData to the cell. Here, we don't need to use the Cell type, because the UITableViewCell object is returned by the dequeueReusableCellWithIdentifier(_:forIndexPath:) method. Phew, such a short implementation, but it's so hard to explain. Then, create a CellConfigurator instance in the items array:
Wait, what’s going on? A compile-time error? Type of expression is ambiguous without more context That's because CellConfigurator is generic, but Swift arrays can only store the same type, we can't simply put CellConfigurator and CellConfigurator into the same array. This is correct, but it's not what we want. Aha, wait a minute, we'll get it done. The Cell type parameter is actually only used when declaring viewData. Therefore, we don't need to specify the actual type of Cell in CellConfigurator. Declare a new non-generic protocol:
Modify CellConfigurator to implement CellConfiguratorType:
Now we can declare the type of items as:
Compilation passed. View Controller Let’s modify the View Controller now. registerCells() can be made simpler:
The good news is that the tableView(_:cellForRowAtIndexPath:) method has also become much simpler:
In order to reuse the View Controller, we still have to do some work. For example, we can make the items modifiable from outside the class. I won’t go into details here. You can refer to the final implementation of the framework and demo on GitHub: ConfigurableTableViewController Conclusion Let's take a look at how the first solution differs from the first two: 1. When we want to add a new cell, we don’t need to modify the View Controller 2. View Controller is type-safe. If we add a type of view data that the cell does not support, we will get a compilation error. 3. No switch statement is required when refreshing the cell. It seems that the third method solves all our problems. I think we have made another step forward. A better solution is often "I searched for it for thousands of times, but when I turned around, I found it was in the dim light." Thanks to Maciej Konieczny and Kamil Kołodziejczyk for reviewing this article. If you liked this article, follow me on Twitter or subscribe to my RSS. |
<<: Android N internal code name - New York Cheesecake
>>: CreditEase Zheng Yun: Sharing on the Practice of Big Data Financial Cloud
How much does Douyin merchant certification cost?...
As an observer of the education and training indu...
Shen Yi's "Reading to Lead the Trend: Fi...
For advertisers , marketing and advertising decis...
Nowadays, everyone is calling for building privat...
On January 31, Muddy Waters Research, a well-know...
Quick Facts 1. Summary of mobile phone vertical s...
1. SB Admin 2 Details & Download 2. Adm...
1. Let me start with three points that I think ma...
The file group template is a powerful Android dev...
In August 2019, I took on a project in which I tr...
Zhihu community is the largest knowledge platform...
Xiaomi has built a company with a valuation of US...
Today I will teach you a method to scan old photo...
What I want to share with you today is the refine...