使用runtime全局修改TableView的Cell的分割线样
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隐藏。