iOS开发造轮子 | UIView及其子类的占位图

作者: 无夜之星辰 | 来源:发表于2017-09-24 22:17 被阅读1630次
iu

这是要封装的


这里展示的仅仅是tableView的占位图。

封装原因

最近重构分类详情页的时候需要实现这个功能,关于这个view我已经封装过几次了,今天我试着回想当初的思路,竟然没什么印象了!对于自己封装过的东西没什么印象原因只有一种:不够简洁。

此事无关代码,关乎优雅

然后我翻看了一下自己以前的写的关于封装这个view的简书,不得不说,挺搓的。。。(每次回头看曾经的代码,都不甚满意。。。)

曾经的问题

1.首先,在基类里实现这个功能就是变相的提升程序的耦合度:我的目的是给tableView添加一个功能,并且是纯粹的添加功能,不需要新类,所以这种情况应该用category。想想MJRefresh,使用多么简单方便,用过一次就难以忘记。

2.连逻辑都繁琐不堪

这是使用方法:

  // 展示无数据占位图
  [self.tableView showEmptyViewWithType:NoContentTypeNetwork];
  // 无数据占位图点击的回调
  self.tableView.noContentViewTapedBlock = ^{
      [SVProgressHUD showSuccessWithStatus:@"没网"];
  };

  // 移除无数据占位图
  [self.tableView removeEmptyView];

如此不堪的使用方法,甚至还要手动移除占位图。

优化

之前只是针对UITableView占位图的封装,这次面向UIView

曾经的问题就是现在要解决的问题,在这之前,先将思路理一理:

封装这个view:
1.至少需要一个参数type来表示是哪种类型的占位图:没网or空数据等等;
2.需要一个回调来处理用户点击重新加载按钮事件。
3.每次展示的占位图只可能有一个。

代码

这是基于UIView的category:

@interface UIView ()

/** 占位图 */
@property (nonatomic, strong) UIView *cq_placeholderView;

@end

@implementation UIView (PlaceholderView)

static void *strKey = &strKey;

- (UIView *)cq_placeholderView {
    return objc_getAssociatedObject(self, &strKey);
}

- (void)setCq_placeholderView:(UIView *)cq_placeholderView {
    objc_setAssociatedObject(self, &strKey, cq_placeholderView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

/**
 展示UIView及其子类的占位图
 
 @param type 占位图类型
 @param reloadBlock 重新加载回调的block
 */
- (void)cq_showPlaceholderViewWithType:(CQPlaceholderViewType)type reloadBlock:(void (^)())reloadBlock {
    // 如果是UIScrollView及其子类,占位图展示期间禁止scroll
    BOOL originalScrollEnabled = NO; // 原本的scrollEnabled
    if ([self isKindOfClass:[UIScrollView class]]) {
        UIScrollView *scrollView = (UIScrollView *)self;
        // 先记录原本的scrollEnabled
        originalScrollEnabled = scrollView.scrollEnabled;
        // 再将scrollEnabled设为NO
        scrollView.scrollEnabled = NO;
    }
    
    //------- 占位图 -------//
    if (self.cq_placeholderView) {
        [self.cq_placeholderView removeFromSuperview];
        self.cq_placeholderView = nil;
    }
    self.cq_placeholderView = [[UIView alloc] init];
    [self addSubview:self.cq_placeholderView];
    self.cq_placeholderView.backgroundColor = [UIColor whiteColor];
    [self.cq_placeholderView mas_makeConstraints:^(MASConstraintMaker *make) {
        make.center.mas_equalTo(self);
        make.size.mas_equalTo(self);
    }];
    
    //------- 图标 -------//
    UIImageView *imageView = [[UIImageView alloc] init];
    [self.cq_placeholderView addSubview:imageView];
    [imageView mas_makeConstraints:^(MASConstraintMaker *make) {
        make.centerX.mas_equalTo(imageView.superview);
        make.centerY.mas_equalTo(imageView.superview).mas_offset(-80);
        make.size.mas_equalTo(CGSizeMake(70, 70));
    }];
    
    //------- 描述 -------//
    UILabel *descLabel = [[UILabel alloc] init];
    [self.cq_placeholderView addSubview:descLabel];
    [descLabel mas_makeConstraints:^(MASConstraintMaker *make) {
        make.centerX.mas_equalTo(descLabel.superview);
        make.top.mas_equalTo(imageView.mas_bottom).mas_offset(20);
        make.height.mas_equalTo(15);
    }];
    
    //------- 重新加载button -------//
    UIButton *reloadButton = [[UIButton alloc] init];
    [self.cq_placeholderView addSubview:reloadButton];
    [reloadButton setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
    [reloadButton setTitle:@"重新加载" forState:UIControlStateNormal];
    reloadButton.layer.borderWidth = 1;
    reloadButton.layer.borderColor = [UIColor blackColor].CGColor;
    [[reloadButton rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
        // 执行block回调
        if (reloadBlock) {
            reloadBlock();
        }
        // 从父视图移除
        [self.cq_placeholderView removeFromSuperview];
        self.cq_placeholderView = nil;
        // 复原UIScrollView的scrollEnabled
        if ([self isKindOfClass:[UIScrollView class]]) {
            UIScrollView *scrollView = (UIScrollView *)self;
            scrollView.scrollEnabled = originalScrollEnabled;
        }
    }];
    [reloadButton mas_makeConstraints:^(MASConstraintMaker *make) {
        make.centerX.mas_equalTo(reloadButton.superview);
        make.top.mas_equalTo(descLabel.mas_bottom).mas_offset(20);
        make.size.mas_equalTo(CGSizeMake(120, 30));
    }];
    
    //------- 根据type设置不同UI -------//
    switch (type) {
        case CQPlaceholderViewTypeNoNetwork: // 网络不好
        {
            NSString *path = [[NSBundle mainBundle] pathForResource:@"无网" ofType:@"png"];
            imageView.image = [UIImage imageWithContentsOfFile:path];
            descLabel.text = @"网络异常";
        }
            break;
            
        case CQPlaceholderViewTypeNoGoods: // 没商品
        {
            NSString *path = [[NSBundle mainBundle] pathForResource:@"无商品" ofType:@"png"];
            imageView.image = [UIImage imageWithContentsOfFile:path];
            descLabel.text = @"一个商品都没有";
        }
            break;
            
        case CQPlaceholderViewTypeNoComment: // 没评论
        {
            NSString *path = [[NSBundle mainBundle] pathForResource:@"沙发" ofType:@"png"];
            imageView.image = [UIImage imageWithContentsOfFile:path];
            descLabel.text = @"抢沙发!";
        }
            break;
            
        default:
            break;
    }
}

@end

使用方法

需要展示占位图的时候直接调用方法:

[self.tableView cq_showPlaceholderViewWithType:CQPlaceholderViewTypeNoComment reloadBlock:^{
    // 按钮点击回调
    [SVProgressHUD showInfoWithStatus:@"重新加载按钮点击"];
}];

细节

1. 对于UIScrollView及其子类,占位图展示期间将它的scrollEnabled设置为NO。

这肯定不是我们想要的效果.gif

但是改变scrollEnabled的时候又不能对原控件造成侵入性,怎么破?

  • 先记录最初的scrollEnabled
// 如果是UIScrollView及其子类,占位图展示期间禁止scroll
BOOL originalScrollEnabled = NO; // 原本的scrollEnabled
if ([self isKindOfClass:[UIScrollView class]]) {
    UIScrollView *scrollView = (UIScrollView *)self;
    // 先记录原本的scrollEnabled
    originalScrollEnabled = scrollView.scrollEnabled;
    // 再将scrollEnabled设为NO
    scrollView.scrollEnabled = NO;
}
  • 移除占位图的时候复原UIScrollView的scrollEnabled
// 复原UIScrollView的scrollEnabled
if ([self isKindOfClass:[UIScrollView class]]) {
    UIScrollView *scrollView = (UIScrollView *)self;
    scrollView.scrollEnabled = originalScrollEnabled;
}

2. 给系统类扩展方法或添加属性都要加上前缀,向SDWebImage学习。

// SDWebImage的方法
[imageView sd_setImageWithURL:nil completed:nil];

3. 图片的加载

不需要一直放在内存里的用imageWithContentsofFile:方法。

4. block

不需要写weakSelf:

[self.view cq_showPlaceholderViewWithType:CQPlaceholderViewTypeNoNetwork reloadBlock:^{
    [SVProgressHUD showSuccessWithStatus:@"有网了"];
    // 直接写self也不会导致内存泄漏
    self.view.backgroundColor = [UIColor redColor];
}];

这里的block是局部变量,跟masonry的block是同一个道理:

- (NSArray *)mas_makeConstraints:(void(^)(MASConstraintMaker *))block {
    self.translatesAutoresizingMaskIntoConstraints = NO;
    MASConstraintMaker *constraintMaker = [[MASConstraintMaker alloc] initWithView:self];
    block(constraintMaker);
    return [constraintMaker install];
}

总结

只有不断的反思和总结才能造出更优雅的轮子。

点击获取demo


2017年10月2日更新

根据评论区Jacob_Liang同学的建议,优化了部分代码,修复了内存泄漏问题,最新代码已更新到GitHub。感谢Jacob_Liang同学及时指出问题。


2017年10月7日更新

根据评论区S_Criminal_7a03同学的建议,新增一个指定占位图frame的方法:

/**
 展示UIView及其子类的占位图,大小可以设置(本质是在这个view上添加一个自定义view)
 
 @param frame 占位图的frame
 @param type 占位图类型
 @param reloadBlock 重新加载按钮点击时的回调
 */
- (void)cq_showPlaceholderViewWithFrame:(CGRect)frame type:(CQPlaceholderViewType)type reloadBlock:(void (^)())reloadBlock;

代码已合并到GitHub,感谢S_Criminal_7a03同学的提议。


2017年10月12日更新

iPhone X的出现断了frame最后的退路,所以我决定使用纯自动布局了。

默认情况下占位图的约束

@weakify(self);
[self.view cq_showPlaceholderViewWithType:CQPlaceholderViewTypeNoNetwork reloadBlock:^{
    @strongify(self);
    [SVProgressHUD showSuccessWithStatus:@"重新加载按钮点击"];
    self.view.backgroundColor = [UIColor redColor];
}];
默认旋转适配.gif

可重新设置占位图的约束

[self.tableView cq_showPlaceholderViewWithType:CQPlaceholderViewTypeNoComment reloadBlock:nil];
self.tableView.cq_placeholderView.backgroundColor = [UIColor redColor];
// 重新设置占位图约束
[self.tableView.cq_placeholderView mas_remakeConstraints:^(MASConstraintMaker *make) {
    make.left.right.mas_equalTo(self.view);
    make.top.mas_equalTo(self.tableView).mas_offset(50);
    make.bottom.mas_equalTo(self.view).mas_offset(-30);
}];
自定义约束的旋转适配.gif

代码已更新到GitHub

相关文章

网友评论

  • 特拉法尔咖:请问,『重新加载按钮』点击事件,还有其他实现方式吗?
    无夜之星辰:用delegate也可以的
  • ISwiftUI:如果是带有tableViewHeaderView或者footerView呢?
    无夜之星辰:那个带frame的方法应该可以满足这个需求吧
  • S_Criminal_7a03:缺少设置显示frame 的地方,当遇到vc嵌套vc,tableView 拥有 headerView的时候 位置设置很麻烦
    无夜之星辰:代码合并到GitHub了,如果你要用更强大的可以看看这个:http://www.jianshu.com/p/beca3ac24031
    无夜之星辰:@S_Criminal_7a03 不错的建议,明天我再添加一个方法:smile:
  • Jacob_LJ:首先感谢你的分享,在你的 demo 中我也找到可以学习的地方,但是你可能稍有欠缺考虑某些东西,比如下面说的情况:

    demo 有内存泄露,情况是:1. 在secondVC 中点击模拟获取数据按钮,让结果首次展示网络不佳情况,即第一次出现占位图后。
    2. 点击返回到 oneVC,此时的 secondVC 并无释放,其他类似。

    还有,你说的与 masonry 的局部变量 block 设计类似而自己的 reloadBlock 不需要使用 weakSelf ,你这个描述是错的,因为你的占位图重新加载按钮的 Action 交给了 RAC 的 block retain 了,所以内部会对 self 进行强引用,这样就出现 retain circle 了。

    至于有时候可以释放 secondVC,完全是因为你主动移除了通过 runtime 添加的 placeholderView 原因,你将它置 nil 之后,这样就打破了循环引用,而且你demo 中,@weakify 和 @stongify 在GCD那个位置用的不恰当,你可以试试就改为 weakSelf,这样就避免了上述的问题了。不能发 gif 图评论,所以描述可能有点欠缺

    最后就因为使用 rac 所以你的这个分类耦合度过高了,使用分类而要引入一个framework,太重型了。

    共同学习,共同进步:smile:
    无夜之星辰:@Jacob_Liang 嗯嗯,相互学习,共同进步:grin:
    Jacob_LJ:多谢你能采纳我的小小建议,能分享文章出来的人都是值得尊敬的人,毕竟写一篇文章不是一件简单的事情。望以后共同进步~:smile:
    无夜之星辰:非常感谢你指出我代码中的问题,这也是我分享的最大收获。

    内存泄漏问题,确实如你所说,我之前考虑有所欠缺,目前已解决并更新到GitHub,欢迎你空闲的时候帮我code reviewer一下😅。

    关于轮子中三方的问题,我其实之前也考虑过,使用三方的目的主要是方便,缺点也很明显:增加了耦合度。最终我的原则是:只使用被大众认可并且被广泛使用的三方,比如我这个分类里面用到的masonry和RAC。不过我个人也是倾向于轮子里面不要有任何三方,这也是我的一个目标吧。

    最后,再次感谢你的细心阅读和有用建议。😄
  • 荔枝lizhi_iOS程序猿:楼主,swift category 没办法 定义属性 UIView *cq_placeholderView 怎么破?
    无夜之星辰:@Sunny_张 我还在学swift:sweat_smile:
  • Zz7777777:- (void)setCq_placeholderView:(UIView *)cq_placeholderView {
    objc_setAssociatedObject(self, &strKey, cq_placeholderView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
    能告诉利用这个runtime 是用来干啥呢
    Zz7777777:@无夜之星辰 https://github.com/LSure/AppPlaceholder
    Zz7777777:@无夜之星辰 我之前还看到过 利用runtime 对tableview的方法进行拦截 然后一句代码实现所有tableview 的无数据占位图 大佬可以学习下
    无夜之星辰:@HelloCoders category中添加属性
  • flowerflower:小伙子有前途。666666
    无夜之星辰:@flowerflower 师姐见笑了:sweat_smile:

本文标题:iOS开发造轮子 | UIView及其子类的占位图

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