Skip to content

Latest commit

 

History

History
136 lines (105 loc) · 5.58 KB

README.md

File metadata and controls

136 lines (105 loc) · 5.58 KB

FCXTableView

在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 = ^(){

};

##下拉刷新、上拉加载更多 自己写了一套下拉刷新、上拉加载更多,只需添加一行代码即可,可以更好的和TableView结合使用.

[_tableView addHeaderWithRefreshHandler:^(FCXRefreshBaseView *refreshView) {

}];

[_tableView addFooterWithRefreshHandler:^(FCXRefreshBaseView *refreshView) {

}];