• UITableViewcell autolayout下动态高度


    项目中最经常使用的一个UI就是UITableView了。iOS7、8进一步优化了复用机制,用起来相当爽。配合Autolayout,适配工作减轻了非常多。

    曾经做适配工作都是在heightForRow里边先计算出来Cell的高度。然后再CellForRow写适配代码。工作量尽管不是非常大,可是非常繁琐。

    相对于这样的写法,假设减去计算height这步,工作量自然降低非常多。首先给出一种我媳妇给提供的方法,这是她做聊天UI时因为过度计算而怒创的方法发火,当时我看到就震惊了,之后我就一直用这种方法害羞



    使用这中方法就可以省反复计算frame的操作,计算一次布局就可以。



    如今IOS7、8都出来这么久了,autolayout也炒的非常热。如今看了下确实方便了非常多。比我在iOS6方便多了。

    前两天看了下autolayout,如今结合tableviewcell试试效果。看了几个demo,发现iOS7跟iOS8不一样,iOS8更加简单。

    第一、iOS7 tableviewcell + autolayout


    使用storyboard简单创建一个project。并加入例如以下约束(不熟悉约束加入方法的可參考我之前的博客autolayout)

    关联project文件,Cell连线,复用标识符等(不熟悉。可參考我之前的博客):




    重头戏来了,代码中改怎样改动:

    cellForRow依旧是我们之前的写法,赋值:

    <span style="color:#993399;">- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
        TestCell *cell = [tableView dequeueReusableCellWithIdentifier:@"testCell" forIndexPath:indexPath];
        NSLog(@"cell create...");
        if (!cell) {
            NSLog(@"cell is nil!");
        }
        cell.ts_label.text = self.data[indexPath.row];
    
        return cell;
    }</span>
    

    重点在于计算height,改动方法例如以下:

    <span style="color:#993399;">- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
    {
        TestCell* cell = [tableView dequeueReusableCellWithIdentifier:@"testCell"];
    
        cell.ts_label.text = self.data[indexPath.row];
        CGSize size = [cell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize];
        NSLog(@"size = %@ %lf",NSStringFromCGSize(size),    cell.ts_label.preferredMaxLayoutWidth
    );
    
        return size.height + 1;
    }
    </span>


    有三点须要注意:

    第一点:不要在heightForRow使用

    - (id)dequeueReusableCellWithIdentifier:(NSString *)identifier forIndexPath:(NSIndexPath *)indexPath

    你会发现进入了死循环。heightForRow->cellforRow->hei...Row

    第二点:使用systemLayoutSizeFittingSize,他会依据约束条件算出来新的size,(所以约束条件一定要准确,非常多BUG都是由于约束有问题)

    它有两种參数:

    UIKIT_EXTERN const CGSize UILayoutFittingCompressedSize NS_AVAILABLE_IOS(6_0);//紧凑,大小适中的最小尺寸

    UIKIT_EXTERN const CGSize UILayoutFittingExpandedSize NS_AVAILABLE_IOS(6_0);//宽松

    第三点:新的size计算的是contentview的size,Cell.size.height = content view.size.height+1,记得加上一个切割线的高度。


    好的到这里看看执行效果:



    咋一看感觉效果实现了,细致观察发现label的高度算的不正确。第三行label不仅仅有三行,所以这个size算的不准。


    经过研究发现,这个时候须要给label一个宽度值,这样才干算出来所须要的高度。

    能够设置label的这个属性来固定宽度。

    // Support for constraint-based layout (auto layout)

    // If nonzero, this is used when determining -intrinsicContentSize for multiline labels

    @property(nonatomic) CGFloat preferredMaxLayoutWidth NS_AVAILABLE_IOS(6_0);


    我们改动heightForRow的方法: 
    - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
    {
        TestCell* cell = [tableView dequeueReusableCellWithIdentifier:@"testCell"];
        cell.ts_label.preferredMaxLayoutWidth = 260;
        cell.ts_label.text = self.data[indexPath.row];
        CGSize size = [cell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize];
        NSLog(@"size = %@ %lf",NSStringFromCGSize(size),    cell.ts_label.preferredMaxLayoutWidth
    );
    
        return size.height + 1;
    }
    

    跑起来看看效果:


    这才像个样子,哇哈哈。

    这个属性也能够在storyboard中设置:







    第二、iOS8 tableviewcell + autolayout


    iOS7的方法已经简单非常多了,只是iOS8更加简单:iOS8仅仅须要两行代码加上约束就可以完毕适配!

    惊恐有木有非常夸张!!

    看方法:

    <span style="color:#993399;">- (void)viewDidLoad {
    </span><span style="color:#663366;">    [super viewDidLoad];
        
        self.data = @[@"hahahahahahahahahahahaha",@"gagagaga",@"lalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalalala",@"wawawawawawawawawawawawawawawawawawawawa",@"papapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapapa",@""];
        self.tableView.estimatedRowHeight = 120;
        self.tableView.rowHeight = UITableViewAutomaticDimension;
    
    }</span>
    


    <span style="color:#663366;">- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
        TestCell *cell = [tableView dequeueReusableCellWithIdentifier:@"testCell" forIndexPath:indexPath];
        NSLog(@"cell create...");
        if (!cell) {
            NSLog(@"cell is nil!");
        }
        cell.ts_label.text = self.data[indexPath.row];
    
        return cell;
    }</span>
    

    看到没有,iOS8设置都不须要heightForRow了。仅仅须要设置下预估高度

    <span style="color:#663366;">estimatedRowHeight</span>
    <span style="color:#663366;"></span><pre name="code" class="objc"><span style="color:#663366;">rowHeight = UITableViewAutomaticDimension</span>

    
    

    iOS7优化了tableview,获取Cell高度不再是一次性所有载入heightForRow。然后再生成Cell。而是先调用:

    - (CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath


    等Cell真正要呈现的时候在调用heightForRow。


    真实的效果呢



    高端大气上档次啊,哇哈哈。

    看看其它机型




    哈哈,就是这么简单,Cell就是配好了!

    !!





    总结一下:iOS7须要注意preferredMaxLayoutWidth的设置,iOS8须要设置estimatedRowHeight、rowHeight = UITableViewAutomaticDimension。更加注意的是约束的准确!

    !。大笑

  • 相关阅读:
    asp.net web生命周期
    图的数据结构1
    最长公共子串
    内部排序

    棋盘覆盖问题
    队列
    矩阵连乘问题
    图的数据结构2
    旅行售货员问题
  • 原文地址:https://www.cnblogs.com/mengfanrong/p/5325618.html
Copyright © 2020-2023  润新知