关于UITableView的HeaderView循环的问题

最近写了一个UITableView内即有section又有row的小项目,顺便在row内添加btn来从新刷新这个UITableView的数据源.
一直到写完加载 显示ui都没有发现有什么问题,包括内存方面,可是在添加点击事件后,用block回调来刷新UITableView的时候发现有bug了,滚动的时候点击会反复的跳动 UITableViewCell 搞得点击btn后 页面闪动很大.
初步猜测有以下一些原因:
  • 调用的方法 -reloadRowsAtIndexPaths:withRowAnimation:出现问题,使用 reloadData方法替换
  • 返回组头的高度- tableView:heightForHeaderInSection:方法出现问题
  • 使用的- tableView: viewForHeaderInSection:设置UITableView的HeaderView的时候错误
    -调用 - (CGFloat)tableView: estimatedHeightForHeaderInSection:方法预估计高度出现错误

关于- tableView:viewForHeaderInSection:
一般这个方法- tableView:viewForHeaderInSection:我会这么调用

- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section {
    UIView *view = [[UIView alloc] init];
    //设置数据源 xxxx.....
    return view;
}

突然感觉这个方法应该向cell那样 有循环的

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellId forIndexPath:indexPath];
 //设置数据源 xxxx.....
    return cell;
}

在实例化cell的时候一般都会实现 -dequeueReusableCellWithIdentifier:forIndexPath:方法来达到循环使用cell的目的.
所以看了相关apple的API发现确实有一个方法 来设置这个UITableViewHeader的,是在iOS6引入的一个类UITableViewHeaderFooterView你需要使用 -initWithReuseIdentifier:方法来使HeaderView达到循环的目的

从写此方法后run发现HeaderView竟然木有了...木有了....-_-!

一定是哪里出了问题,谷歌搜索后说需要在实例化的 HeaderView内种重写 ``` - (instancetype)initWithReuseIdentifier:reuseIdentifier


  • (instancetype)initWithReuseIdentifier:(NSString *)reuseIdentifier{

    self = [super initWithReuseIdentifier:reuseIdentifier];
    if (self) {
    return [[[NSBundle mainBundle] loadNibNamed:@"headerView" owner:nil options:nil] lastObject];
    }
    return self;
    }

重写```- (UIView *)tableView:viewForHeaderInSection:```方法

pragma mark 返回每组的头部视图

  • (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section
    {
    static NSString *viewIdentfier = @"headView";
    headerView *headerViews = [tableView dequeueReusableHeaderFooterViewWithIdentifier:viewIdentfier];
    if(!headerViews){
    headerViews = [[headerView alloc] initWithReuseIdentifier:viewIdentfier];
    }
    //设置数据源 xxxx.....
    //去除颜色警告..
    //UITableViewHeaderFooterView has been deprecated. Please use contentView.backgroundColor instead 将背景颜色设置成默认即可去除 在xib内
    headerViews.contentView.backgroundColor = [UIColor blueColor];
    return headerViews;
    }
在写完后 ```run```有报警告 ``` UITableViewHeaderFooterView has been deprecated. Please use contentView.backgroundColor instead ``` 需要从新设置HeaderView的头部颜色要实现```contentView```的方法,直接在xib内修改颜色不起作用哈,把xib的颜色设置成默认,这个警告就消失了,这个是个坑.不过,在xib内在添加一个view改变view颜色也可用达到改变背景颜色的目的.

回归正题,可悲的事情点击btn刷新UITableViewCell后,cell依然乱跳,
重写 ``` - tableView:heightForHeaderInSection:``` 方法....好吧,跟这个方法没关系,略过一些废话.
 
就最后一个了 ```- (CGFloat)tableView:estimatedHeightForHeaderInSection:```
这个方法是iOS7出来的,问题确实出现在这里,在点击cell内的btn调用刷新的时候 高度应该是已经算好了的,不应该在预估一个定值的高度,所以在block回调内当点击btn后 这个方法应该返回真实的高度,而不是预估的高度.
注意:```self.estimatedHeightForRow```是我定义的属性,在cell的block回调内设置成YES;

  • (CGFloat)tableView:(UITableView *)tableView estimatedHeightForHeaderInSection:(NSInteger)section{
    // 去除来回跳转的bug-->
    //在刚刚进入这个控制器的时候 这个 预算高度被激活,当点击cell刷新的时候这个 预算返回 指定的高度
    if (self.estimatedHeightForRow) {
    NSString * str = [self strE:list_[section]];
    CGFloat textH = [self autoCalculateWidthOrHeight:MAXFLOAT width:[UIScreen mainScreen].bounds.size.width - 20 fontsize:15 content:str] + 45;
    return textH;
    }else{
    return 100;
    }
    }

//根据str的内容计算HeaderView的高度
-(float)autoCalculateWidthOrHeight:(float)height
width:(float)width
fontsize:(float)fontsize
content:(NSString*)content
{
//计算出rect
CGRect rect = [content boundingRectWithSize:CGSizeMake(width, height)
options:NSStringDrawingUsesLineFragmentOrigin | NSStringDrawingUsesFontLeading
attributes:@{NSFontAttributeName:[UIFont systemFontOfSize:fontsize]} context:nil];

//判断计算的是宽还是高
if (height == MAXFLOAT) {
    return rect.size.height;
}
else{
    return rect.size.width;
}

}

补充:
有得时候,你会感觉 这个根据文字算高度的方法 ```-boundingRectWithSize: options:attributes: context:```这个方法算的不准,有那么几个原因
1. 参数 ```options: ```不是```NSStringDrawingUsesLineFragmentOrigin | NSStringDrawingUsesFontLeading```
2. attributes:参数错误
3. 在xib内设置的样式不是```System```样式的

![xib_lable_Image.png](http://upload-images.jianshu.io/upload_images/1447422-b9876f9529ebe8a6.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

具体的代码还在整理中.......


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容