A simple view model for building organized and scalable TableViews.
Apple provides the UITableViewDataSource & UITableViewDelegate to implement the tableview UI, but a big problem is that the UI logic exists in multiple places and there will be lots of duplicate code, especially those complex tableview UIs with different types of cells and dynamic logic.
e.g. Let's take a look at a store management UI like this:
2 kinds of users will use this UI, manager and employee. Manager can see all above rows, while employee can only see some of them, like
In the traditional way, we may implement the UI like this:
#pragma mark - UITableViewDataSource
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
switch (self.type) {
case MemberTypeEmployee:
return 3;
break;
case MemberTypeManager:
return 4;
break;
default:
return 3;
break;
}
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
if (section == 0) {
if (self.type == MemberTypeEmployee) {
return 1; // only store info
} else {
return 2; // store info and goods entry
}
} else if (section == 1) {
if (self.type == MemberTypeEmployee) {
return 2; // order list
} else {
return 3; // advanced settings...
}
} else if (section == 2) {
if (self.type == MemberTypeEmployee) {
return 1; // about
} else {
return 3; // store income and withdraw
}
} else {
return 1; // about
}
}
...
There will be more if else code in the rest datasrouce & delegate methods, especially in the tableView:cellForRowAtIndexPath: method. Take a look at the code in BadTableViewController in the demo for more details.
We can see the problem here. There are so many hardcode things and duplicate code. What we need is to use a model to hold all the logic and only use the model in these delegate & datasource methods. That is what the SigmaTableViewModel does.
SigmaTableViewModel provides the YZSTableViewModel as the view model and a depth 2 array sectionModelArray in it. The 1st level of the array is the section, and the 2nd level is the row. So what we need is to construct this array and ask tableview to use it, like this:
- (void)viewDidLoad {
[super viewDidLoad];
self.viewModel = [[YZSTableViewModel alloc] init];
self.tableView.delegate = self.viewModel;
self.tableView.dataSource = self.viewModel;
[self initViewModel];
[self.tableView reloadData];
}
- (void)initViewModel {
[self.viewModel.sectionModelArray removeAllObjects];
[self.viewModel.sectionModelArray addObject:[self storeInfoSection]];
if (self.type == MemberTypeManager) {
[self.viewModel.sectionModelArray addObject:[self advancedSettinsSection]];
}
[self.viewModel.sectionModelArray addObject:[self incomeInfoSection]];
[self.viewModel.sectionModelArray addObject:[self otherSection]];
}
- (YZSTableViewSectionModel*)storeInfoSection {
YZSTableViewSectionModel *sectionModel = [[YZSTableViewSectionModel alloc] init];
...
// store info cell
YZSTableViewCellModel *cellModel = [[YZSTableViewCellModel alloc] init];
[sectionModel.cellModelArray addObject:cellModel];
cellModel.height = 80;
cellModel.renderBlock = ^UITableViewCell *(NSIndexPath *indexPath, UITableView *tableView) {
...
};
if (self.type == MemberTypeManager) {
// product list cell
YZSTableViewCellModel *cellModel = [[YZSTableViewCellModel alloc] init];
[sectionModel.cellModelArray addObject:cellModel];
cellModel.renderBlock = ^UITableViewCell *(NSIndexPath *indexPath, UITableView *tableView) {
...
};
cellModel.selectionBlock = ^(NSIndexPath *indexPath, UITableView *tableView) {
[tableView deselectRowAtIndexPath:indexPath animated:YES];
...
};
}
return sectionModel;
}
...
Take a look at the code in the GoodTableViewController in the demo for more details.
If we want to move the Withdraw row to the 3rd row in that section, what do we need to do in the traditional way? And how about in SigmaTableViewModel way? What if we what to hide the Income row for employee?
2 steps to use the SigmaTableViewModel.
- Create an instance of the YZSTableViewModel and use it as the tableview's delegate and datasource.
- Construct the sectionModelArray for the view model instance.
- Add
pod 'SigmaTableViewModel'
to your Podfile. - Run
pod install
orpod update
.
- Download all the files in the
Lib
subdirectory. - Add the source files to your Xcode project.
- SigmaTableViewModel only provides some frequently used functions of UITableViewDataSource & UITableViewDelegate. If you needs more functions, you can subclass YZSTableViewModel and implement those functions.
- If the code in those cell's blocks can be reused, we can put them in some methods and only invoke these methods in those blocks.
- The block is used frequently in the view model so we have to be careful about the retain cycle. Always use weak-strong dance for safe. In the demo, we use YZWeak & YZStrong macros to simplify the weak-strong dance code.