UITableViewCell自适应图片高度

作者: WenBo丨星空灬 | 来源:发表于2017-12-29 17:05 被阅读0次

    一、问题描述

    最近在做公司项目的时候,遇到了前端上传多张图片到七牛服务器后,前端在列表展示图片适配的问题。一开始我设置的固定高度,因为图片尺寸不是固定的,会出现不同程度的拉伸情况,用户体验也很不好。通过设置图片视图contentMode这个属性,虽然能保证图片不被拉伸,但图片会出现显示不全的问题。

    二、解决思路

    1、后台返回每张图片的宽高,根据比例去计算图片的高度

    2、获取到image,image有size这个属性,可以拿到宽高,根据比例去计算图片的高度

    三、最终采用方案

    第一种方案是可以实现图片的适配问题,我采用了第二种。图片都是网络加载的,要想拿到image,就必须要将图片下载下来,这样就可以获取到图片的尺寸。下面贴出我实现的关键步骤吧。

    1、工程导入SDWebImageMasonry,导入相关头文件。

    2、SDWebImage下载图片,并计算出图片高度,将高度缓存到字典。自定义cell用Masonry设置好上下左右约束,也可以是xib。

    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
        ImageViewCell *cell = [tableView dequeueReusableCellWithIdentifier:kIdentifier forIndexPath:indexPath];
        cell.selectionStyle = UITableViewCellSelectionStyleNone;
        NSString *url = self.imageUrlArray[indexPath.row];
        [cell.autoImageView sd_setImageWithURL:[NSURL URLWithString:url] placeholderImage:nil options:SDWebImageRetryFailed completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, NSURL *imageURL) {
            if (image.size.height) {
                /**  < 图片宽度 >  */
                CGFloat imageW = [UIScreen mainScreen].bounds.size.width - 2 * 15;
                /**  <根据比例 计算图片高度 >  */
                CGFloat ratio = image.size.height / image.size.width;
                 /**  < 图片高度 + 间距 >  */
                CGFloat imageH = ratio * imageW + 15;
                /**  < 缓存图片高度 没有缓存则缓存 刷新indexPath >  */
                if (![[self.heightDict allKeys] containsObject:@(indexPath.row)]) {
                    [self.heightDict setObject:@(imageH) forKey:@(indexPath.row)];
                    [self.tableView beginUpdates];
                    [self.tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone];
                    [self.tableView endUpdates];
                }
            }
        }];
        return cell;
    }
    

    注意

    • 字典key也可以是图片地址url
    • 因为图片下载是异步的,要在下载完成图片之后,如果字典没有缓存当前indexPath高度,需要手动去刷新一次indexPath(最开始的时候,我没有根据key去判断是否需要刷新,当cell滑动的时候,会不停的调用cellForRowAtIndexPath这个方法,会不停的刷新,这样也非常耗费性能)。

    3、cell高度代理方法返回缓存的图片高度

    - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
        return [[self.heightDict objectForKey:@(indexPath.row)] floatValue];
    }
    

    到这里,就已经实现了图片自适应高度的需求了,具体详情,请看测试UITableViewCellAutoImageHeight。实现这个需求,自己也折腾了一段时间,上面的方法现在也能满足项目的需求,但是在UI效果上感觉还是有一些不完美之处。如果大神有更好的实现方案和优化方法,希望能多多交流学习。如果以后有更好的实现方案,我也会在这篇文章中记录下来。

    四、结语

    2017年还剩两天,这篇文章也是今年写的最后一篇文章吧,写文章的初衷也是为了记录自己学习成长的点滴,同时也希望能够帮助到需要的人。2017就要结束了,自己的期望也未达到,希望在即将到来的2018年,自己更加努力,能够实现自己的期望与目标。奋斗吧,少年!

    相关文章

      网友评论

        本文标题:UITableViewCell自适应图片高度

        本文链接:https://www.haomeiwen.com/subject/jgfwgxtx.html