【前言】

  在使用华尔街见闻 app 时,看到它的 tableVeiw 上的 cell 具有很好的展开与收缩功能。于是自己想了一下实现,感觉应该挺简单的,于是心痒痒写个 demo 实现一波。华尔街见闻 app 上的效果如下:

  

【本 demo 实现的效果图】

    

【思路】

  由它的效果图可以观察出,cell 上默认显示文字多于 4 行时省略,点击时文字全部展现,cell 也同时适应文字的高度。

  1. label 行数可以用 numberOfLines 属性来控制,改变它就可以改变文字的高度。

  2. cell 的高度需要变化,但是如果根据文字收缩时算一下高度,展开时又给 cell 一个高度这样会很麻烦,因为这样要去精确计算出文字高度,上下间距。所以这里我想到的是利用 tableView 估算 cell 高度的机制,cell 内的文字用 label 的约束将 cell "撑满"。

  3. 用一个 model 对应一个 cell 来防止复用数据错乱,然后展开与收缩用 tableView 的刷新动画就好了。

   4. 实际实现过程中遇到一些小坑,重点在处理动画流畅思路上。

【代码实现】

  1. 开启 tableView 估算机制

- (HomeView *)homeView
{
if (!_homeView) {
_homeView = [[HomeView alloc] init];
_homeView.backgroundColor = [UIColor whiteColor];
_homeView.tableView.dataSource = self;
_homeView.tableView.delegate = self;
_homeView.tableView.estimatedRowHeight = ;
[_homeView.tableView registerClass:[HomeCell class] forCellReuseIdentifier:NSStringFromClass([HomeCell class])];
[self.view addSubview:_homeView];
}
return _homeView;
}

  2. label 默认 numberOfLines 设置

- (UILabel *)contentL
{
if (!_contentL) {
_contentL = [[UILabel alloc] init];
_contentL.frame = CGRectMake(, , self.contentView.bounds.size.width, );
_contentL.lineBreakMode = NSLineBreakByTruncatingTail;
_contentL.numberOfLines = ;
[self.contentView addSubview:_contentL];
}
return _contentL;
}

  3. cell 数据设置 (通过 model 内 isOpen 来防止 cell 复用数据发生错乱)

- (void)setModel:(HomeModel *)model
{
_model = model;
self.contentL.text = model.title; if (model.isOpen) {
self.contentL.numberOfLines = NSIntegerMax;
} else {
self.contentL.numberOfLines = ;
}
}

  4. 点击时 tableView 动画刷新

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
HomeCell *cell = [tableView cellForRowAtIndexPath:indexPath];
cell.model.isOpen = !cell.model.isOpen; [tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationFade];
}

【细节优化】

  上面的实现看似美好且不费力,但实际多滑两下,多点一些不同高度的 cell,会发现它的展开动画明显不流畅,会出现一点抖动状况。如下:

  

  看到这里,展开和收缩动画在这个 cell 上简直抽风了,有种突然截断了的感觉。这是肯定不能忍的。

  优化分析:

  tableView 动画不会有问题,是很正常的刷新对应的 cell。唯一可能出问题的就是 cell 的估算机制。因为 cell 我们默认给的高度是 80,所以当 cell 实际高度与默认高度相差太大时,就会出现这种问题了。

  解决方案:  

  我们还是采用 cell 高度的估算机制,但要让系统尽可能估计得准确,减少它调整高度带来的动画不流畅。那就是利用 tableView 返回估算高度的代理方法,我们先给它算好一个高度,并且这个高度肯定基本准确。(注意: 提前在拿到 model 数据时就算好 cell 高度存放在 model 中,而不是在 tableView 代理方法中去计算 cell 高度,因为代理方法调用太频繁,那里做一些简单取值操作就好,不然当数据量大时影响 tableView 滑动性能。)

  1. 在 model 获取数据时就算好 cell 数据的高度。在 HomeModel 加一个 cellHeight 属性,并重写 setTitle: 方法。如下:

- (void)setTitle:(NSString *)title
{
_title = title; CGFloat w = [UIScreen mainScreen].bounds.size.width - ;
CGFloat h = MAXFLOAT;
NSDictionary *dict = @{NSFontAttributeName: [UIFont systemFontOfSize:]};
CGFloat height = [title boundingRectWithSize:CGSizeMake(w, h) options:NSStringDrawingUsesLineFragmentOrigin attributes:dict context:nil].size.height;
_cellHeight = height + ;
}

  2. 在控制器中加上 tableView 返回估算高度的代理方法设置如下:

- (CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath
{
HomeModel *model = self.testDataArray[indexPath.row];
return model.cellHeight;
}

  这样再运行,无论怎么滑怎么点,都十分完美。

最新文章

  1. Struts2登录小例子
  2. xcode 8   去除无用打印信息
  3. php常见问题
  4. HDU3635Dragon Balls(并查集)
  5. iOS开发--图片处理
  6. JSOI2014第三轮总结
  7. 让asp.net web api同时支持[AcceptVerbs("GET","POST")]
  8. 【转载】TCP协议疑难杂症全景解析
  9. 《BackboneJS框架的技巧及模式》(4)完结篇
  10. Android读取url图片保存及文件读取
  11. spring与axis2整合发布webservice
  12. JAVA集合差异
  13. Javascript经典实例 - 字符串
  14. JAVA设计模式初探之装饰者模式
  15. 安装mysql5.5.28的步骤 2017.6.27
  16. jsp 异步处理
  17. PL/SQL执行计划查看
  18. AWR不能自动生成快照
  19. idea打开项目没有文件目录
  20. Django makemigrations 不行时

热门文章

  1. 3.MySQL的架构介绍
  2. linux——实际工作中如何使用linux
  3. JS继承的原理、方式和应用
  4. okhttp初识拦截器
  5. c语言啊
  6. Spring源码解读(一)
  7. CF487E Tourists[圆方树+树剖(线段树套set)]
  8. Invalid argument: Key: label. Data types don't match. Data type: int64 but expected type: float
  9. service worker在移动端H5项目的应用
  10. BZOJ 4147: [AMPPZ2014]Euclidean Nim (分类讨论博弈神题)