使用runtime全局修改TableView的Cell的分割线样

2017-06-21  本文已影响0人  小志Shannon

UITableView的分割线样式默认是使用一个View,使用view的颜色来作为分割线,高度只要一个像素。

项目遇到这么一个问题,由于是新来的一个UI设计师,竟然把cell的分割线都设计了一套样式,由于之前一直是使用系统原生的cell分割线,那么问题就来了,整个工程的cell分割线都要去修改(蛋蛋的忧伤ing),想到的解决方案有

1、一个将UITableView的separatorStyle属性设置为UITableViewCellSeparatorStyleNone,然后在cell内部自己用一个imageview放到cell的最底部,然后设置高度是1个像素。

2、写一个继承自UITableViewCell的基类,所有的Cell都继承自这个基类,基类里面统一添加一个imageview在最底部。

3、是不是可以考虑用runtime来处理这个问题呢?

分析了一波,作为一个程序员,第一要素,能偷懒的地方绝对去偷懒。所以果断最后考虑第一个PlanA(项目里面没有三十就要四十个cell,一个一个改,吐血ing),

那么分析一波PlanB,基类,要重写几个方法有

- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(nullableNSString*)reuseIdentifier;

- (void)awakeFromNib; 

这几个方法,然后统一调用一个方法创建一个imageview放到self.contentView上,接着为了防止被别的视图盖住,可以将这个imageview放到最底层,[self.contentView bringSubviewToFront:imageview]; ,然后将所有的UITableViewCell的类的基类都改成刚刚创建的类。

似乎好像可以实现了,紧接着另一个问题出现了。。如果UITableView不需要分割线呢? 基类设置了分割线,那么所有的cell都一定有分割线,既然不要分割线,那么就可以将基类提供一个方法显示隐藏分割线,属性或者方法都可以。

那么运行后另一个问题来了,系统原生分割线同样存在+你自己自定义的分割线,分割线就是双重的了,并不是我们想要的想过。那么我们就可以在Cell的基类的layoutSubViews中写self.separatorInset = UIEdgeInsetsMake(0, 0, self.contentView.frame.size.height, 0); 这样就可以隐藏系统原生的分割线而留下我们自定义的分割线了。

似乎事情就这么轻而易举的解决了,那么还有更好的方法么? 例如只要一个类,其他的什么代码都不用动就能完成呢?

那我们接着往下思考,既然要这样来处理问题的话,似乎只能使用runtime来处理问题了。。。

查看一下Xcode的,可以看到布局上有一个叫做@“_UITableViewCellSeparatorView”的东东

点击了发现就是分割线那个view,那么就好办了,可不可以对这个控件做点手脚呢? 然而苹果的API并没有暴露这个控件可修改,只能改变

这些属性。。。改改颜色,样式啥的。。。并不是我们想要的。那么想想,所有控件最终都要走layoutSubviews方法,在这里可不可以遍历出这个控件做处理呢?肯定可以的,为了方便管理所有的cell的样式,使用Method Swizzling来替换layoutSubviews方法,统一处理这个问题。代码如下:

1、创建一个UITableViewCell的分类

#import <UIKit/UIKit.h>

@interface UITableViewCell (Extension)

@end

然后在.m文件里面写:

#import "UITableViewCell+Extension.h"

#import <objc/runtime.h>

@implementation UITableViewCell (Extension)

// 为了只运行一次,可以用单例来处理

+ (void)load{

     static dispatch_once_t onceToken;

    dispatch_once(&onceToken, ^{

        SEL originalSelector = @selector(layoutSubviews);//原始方法

        SEL swizzledSelector = @selector(xdd_layoutSubViews);// 要替换的方法

        Method originalMethod = class_getInstanceMethod([UITableViewCell class], originalSelector);

       Method swizzledMethod = class_getInstanceMethod([UITableViewCell class], swizzledSelector);

       method_exchangeImplementations(originalMethod, swizzledMethod);

    });

}

// 用此方法替换系统的layoutSubviews方法

- (void)xdd_layoutSubViews{

      [self xdd_layoutSubViews]; // 先让所有控件布局完成后再做操作(至于为什么调用自己,可以百度一下)

      if([[view.classForCoder description] isEqualToString:@"_UITableViewCellSeparatorView"]) {

            if(view.backgroundColor) {//如果tableView设置了separatorStyle为UITableViewCellSeparatorStyleNone的时候,backgroundColor的颜色为空(null)

                  view.backgroundColor= [UIColorex_separatorColor]; // 修改颜色

            }

      }

}

@end

因为load方法系统自己会调用,所以只要这个类存在于工程中,什么操作都不需要了,不影响系统原生的cell分割线方法,

整个工程只要用到UITableViewCell的地方,分割线就都变化了,而且不影响你在任何一处使用separatorStyle设置为none隐藏。

上一篇下一篇

猜你喜欢

热点阅读