FCXTableView为Controller瘦身的UITableview
在iOS开发的过程中,UITableview是使用频率很高的控件之一,今天写的优化方法不是关于性能优化方面的,主要从为Controller瘦身方面考虑的。在使用tableView的时候不可避免的要谈到tableView的delegate和dataSource两个代理,我们经常会把这两个代理赋给Controller,在Controller里面我们会实现它的几个代理方法,最常见的有以下几个:
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section; - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath; - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath;
这里会在Controller里面产生许多不必要的代码,下面就从这两个代理方法入手,来为Controller瘦身。
delegate、dataSource从Controller中去掉,交给Tableview自己处理
在FCXTableView中将Tableview的delegate、dataSource付给自己。
- (void)fcx_setUp { self.delegate = self; self.dataSource = self; }
为了实现相应的代理方法,Tableview必须要拿到数据源,考虑到Tableview有分组和不分组两种情况,这里增加了两个属性,其中dataArray是只有一组的情况(使用dataArray时会自动把dataArray放到一个数组里然后再赋值给groupArray),groupArray是多组时用到的,如果项目中不需要分组情况时groupArray是多余的,但为了考虑兼容问题还是加上了。
@property (nonatomic, strong) NSMutableArray *groupArray; @property (nonatomic, strong) NSMutableArray *dataArray; - (void)setGroupArray:(NSMutableArray *)groupArray { NSAssert(groupArray, @"groupArray必须是数组类型"); if (![groupArray isKindOfClass:[NSArray class]]) { return; } if (_groupArray != groupArray) { _groupArray = groupArray; [self reloadData]; } } - (void)setDataArray:(NSMutableArray *)dataArray { NSAssert(dataArray, @"dataArray必须是数组类型"); if (![dataArray isKindOfClass:[NSArray class]]) { return; } self.groupArray = [[NSMutableArray alloc] initWithObjects:dataArray, nil]; }
在拿到数据源之后就可以实现代理方法了,后面解释为什么判断self.groupArray.count == 0和setDataModel:
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView { if (self.groupArray.count == 0) {//无数据时 return 1; } return self.groupArray.count; } - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { if (self.groupArray.count == 0) {//无数据时 return 1; } return [self.groupArray[section] count]; } - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { if (self.groupArray.count == 0) {//无数据时 return 300; } return 44; } - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { if (self.groupArray.count == 0) {//无数据时 return self.noDataCell; } NSAssert([self.groupArray[indexPath.section] isKindOfClass:[NSArray class]], @"groupArray中的数据必须是数组类型"); UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:self.cellIdentifier forIndexPath:indexPath]; if (self.groupArray.count > indexPath.section && [self.groupArray[indexPath.section] count]) { id dataModel = [self.groupArray[indexPath.section] objectAtIndex:indexPath.row]; //这里的setDataModel:是更新cell数据模型的方法,可自行定义,可参考FCXTableViewCell if ([cell respondsToSelector:@selector(setDataModel:)]) { [cell performSelectorOnMainThread:@selector(setDataModel:) withObject:dataModel waitUntilDone:NO]; } } return cell; }
将数据和Cell关联。
在拿到数据源groupArray后要和展示的Cell进行关联,在定义Cell的时候每个Cell加一个dataModel的属性,默认会调用setDataModel:(上面提到的)这个方法,可以在这个方法里进行数据的处理。
- (void)setDataModel:(NSString *)dataModel { }
将点击某行Cell的代理方法用Block替代
@property (nonatomic, copy) FCXDidSelectRowBlock didSelectRowBlock; _tableView.didSelectRowBlock = ^(NSIndexPath *indexPath, id data) { };
无数据展示优化
用Tableview展示数据的时候就会遇到没有数据或者网络请求失败等情况,需要给用户展示一个当前的无数据状态(上面提到的self.groupArray.count == 0,这个用来判断无数据情况),好点的做法是在设计的时候这里能够用一个通用的模板展示样式,不过这里支持自定义展示样式并支持无数据状态的点击响应事件(noDataActionBlock用Block方式实现),只需传入你定义展示样式的noDataViewClass即可(具体可参考Demo)。
@property (nonatomic, strong) Class noDataViewClass; @property (nonatomic, copy) FCXNoDataActionBlock noDataActionBlock; _tableView.noDataActionBlock = ^(){ };