美文网首页iOS系列(LJ)iOS 技术文集iOS学习笔记
UITableView设置全屏分隔线的几种方法比较

UITableView设置全屏分隔线的几种方法比较

作者: 苹果API搬运工 | 来源:发表于2016-04-11 21:18 被阅读15926次

一般TableView设置全屏分隔线有下面三种方法

1.自定义cell,手动添加分割线

  • 隐藏自带的
    tableView.separatorStyle = UITableViewCellSeparatorStyleNone;

  • 可以通过addSubview的方式添加一条分割线;也可以自绘分割线。

// 自绘分割线
- (void)drawRect:(CGRect)rect
{
    CGContextRef context = UIGraphicsGetCurrentContext();
    
    CGContextSetFillColorWithColor(context, [UIColor whiteColor].CGColor);
    CGContextFillRect(context, rect);
    
    CGContextSetStrokeColorWithColor(context, [UIColor colorWithRed:0xE2/255.0f green:0xE2/255.0f blue:0xE2/255.0f alpha:1].CGColor);
    CGContextStrokeRect(context, CGRectMake(0, rect.size.height - 1, rect.size.width, 1));
}

2.重写cell的setFrame方法,高度-1,露出背景色

- (void)setFrame:(CGRect)frame
{
    frame.size.height -= 1;
    // 给cellframe赋值
    [super setFrame:frame];
}
  • 取消系统的分割线
  • 设置tableView背景色为分割线颜色
 // 取消系统分割线
    self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone;
    // 设置tableView背景色
    self.tableView.backgroundColor = [UIColor colorWithWhite:215 / 255.0 alpha:1];

3.利用系统属性设置(separatorInset, layoutMargins)共需添加三句代码:

  • 对tableView的separatorInset, layoutMargins属性的设置
-(void)viewDidLoad {
    [super viewDidLoad];
    //1.调整(iOS7以上)表格分隔线边距
    if ([self.tableView respondsToSelector:@selector(setSeparatorInset:)]) {
        self.tableView.separatorInset = UIEdgeInsetsZero;
    }
    //2.调整(iOS8以上)view边距(或者在cell中设置preservesSuperviewLayoutMargins,二者等效)
    if ([self.tableView respondsToSelector:@selector(setLayoutMargins:)]) {
        self.tableView.layoutMargins = UIEdgeInsetsZero;
    }
}
  • 对cell的LayoutMargins属性的设置
对cell的设置可以写在cellForRowAtIndexPath里,也可以写在willDisplayCell方法里

-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *ID = @"cell";
    FSDiscoverSpecialCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
    if (cell == nil) {
        cell = [[FSDiscoverSpecialCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID];
    }

   //2.调整(iOS8以上)tableView边距(与上面第2步等效,二选一即可)
    if ([cell respondsToSelector:@selector(setPreservesSuperviewLayoutMargins:)]) {
        cell.preservesSuperviewLayoutMargins = NO;
    }
   //3.调整(iOS8以上)view边距
    if ([cell respondsToSelector:@selector(setLayoutMargins:)]) {
        [cell setLayoutMargins:UIEdgeInsetsZero];
    }
    return cell;
}

三种方法优缺点比较:

  • 方法1是比较好用的,但是有些情况下系统自带的cell就足够用了,仅仅为了分隔线却还必须再自定义cell,添加一个view,设置背景颜色和frame,又显得麻烦;

  • 方法2比较取巧,但是也需要自定义cell,在某些情况下不允许改变tableView的背景色,使用场景有限;

  • 方法3不需要自定义cell,对系统(iOS7,iOS8以上)做个简单判断即可.可惜网上很多文章写的不对,很多人不会正确使用,有些会用的人也说不清楚原理,只管复制粘贴.
    比如网上流传的一般是这样,需要四步,虽然真的管用,但多了一步[cell setSeparatorInset:UIEdgeInsetsZero];而且原理也没讲,估计是某大神写的,根本不屑于过多解释,让我用起来很郁闷,网上流传代码:

首先在viewDidLoad方法中加上如下代码:
-(void)viewDidLoad {
    [super viewDidLoad];
    if ([self.tableView respondsToSelector:@selector(setSeparatorInset:)]) {
    [self.tableView setSeparatorInset:UIEdgeInsetsZero];
    }
    if ([self.tableView respondsToSelector:@selector(setLayoutMargins:)]) {
    [self.tableView setLayoutMargins:UIEdgeInsetsZero];
}
然后在willDisplayCell方法中加入如下代码:
- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath
{
    if ([cell respondsToSelector:@selector(setSeparatorInset:)]) {
        [cell setSeparatorInset:UIEdgeInsetsZero];
   }
    if ([cell respondsToSelector:@selector(setLayoutMargins:)]) {
        [cell setLayoutMargins:UIEdgeInsetsZero];
    }
}

其实关于分隔线不能全屏的原理,苹果官方在文件中已经说明了,可以去看一下


在iOS7之前系统默认就是全屏的,iOS7时UITableView多了separatorInset属性,可在UITableView的头文件中查看,如下:

@property (nonatomic) UIEdgeInsets separatorInset NS_AVAILABLE_IOS(7_0) UI_APPEARANCE_SELECTOR; 
// allows customization of the frame of cell separators

iOS7时只要设置该属性为UIEdgeInsetsZero就没有问题了.


iOS8之后仅仅完成以上设置就不行了,仔细查看后发现iOS8的UIView
的头文件里又多了个layoutMargins属性,并有官方注释

@property (nonatomic) UIEdgeInsets layoutMargins NS_AVAILABLE_IOS(8_0);
/*
 -layoutMargins returns a set of insets from the edge of the view's bounds that denote a default spacing for laying out content.
 If preservesSuperviewLayoutMargins is YES, margins cascade down the view tree, adjusting for geometry offsets, so that setting the left value of layoutMargins on a superview will affect the left value of layoutMargins for subviews positioned close to the left edge of their superview's bounds
 If your view subclass uses layoutMargins in its layout or drawing, override -layoutMarginsDidChange in order to refresh your view if the margins change.
 */
大意是说:layoutMargins是view的bounds的边距,用来调整内容默认边距

如果preservesSuperviewLayoutMargins属性是YES,那么设置父控件的layoutMargins边距,
就会影响所有子控件的相对于父控件bounds的layoutMargins边距

如果你的view的子类在布局或者绘图中使用了layoutMargins属性,需要重写-layoutMarginsDidChange 方法,
以便当边距改变时能刷新你的view

正是因为layoutMargins是UIView的新增属性,tablet和cell作为UIView的子类都有这个属性,所以相比较iOS7系统,iOS8之后就多了两步,必须同时再对tableView和cell的layoutMargins属性进行处理,才能让分隔线真正全屏.

同时官方注释中对preservesSuperviewLayoutMargins(意即:维持父控件的布局边距)属性的说明,也正好能说明网上另一种方法不设置self.tableView.layoutMargins = UIEdgeInsetsZero;而是设置cell.preservesSuperviewLayoutMargins = NO;为什么也能起作用

弄清楚了这些原理,就可以更好的记忆和使用这些方法,不用每次都去旧代码查找或者去百度了.

说到了最后,不知道大家有没有觉得影响分隔线全屏的元凶layoutMargins属性 稍微有点眼熟呢?其实它在另一个地方也做了不少恶,就在storyboard中:

QQ20150413-000@2x.png

PS:附效果图如下:


设置之前效果图:

QQ20150411-111@2x.png
设置完第1步self.tableView.separatorInset = UIEdgeInsetsZero;后效果图:
QQ20150411-222@2x.png
设置完第2步self.tableView.layoutMargins = UIEdgeInsetsZero;后效果图:
QQ20150411-333@2x.png
设置完第3步cell.layoutMargins = UIEdgeInsetsZero;后效果图:
QQ20150411-444@2x.png

相关文章

网友评论

  • 溪石iOS:给楼主专研精神点赞
  • Lotheve:博主,我在iOS10.1中尝试只设置[tableView setSeparatorInset:UIEsgeInsetsZero]即可达到让分割线顶边的目的啊,并没有设置任何layoutMargins相关的,请问这是怎么回事呢?有点晕了
    薛定谔的熊:经测试你这个在iOS8、9上都不会达到目的,10、11上可行,所以为了兼容8、9加上cell.layoutMargins = .zero 即可
    跑神的心:这种方法有 BUG,当你的 cell 滑到底,然后重新滑上来的时候,上面部分分割线会消失
    可惜卟是迩:我试了下,真可以,谢谢
  • e02382c60b88:self.tableView.separatorInset = UIEdgeInsetsZero;
    if ([cell respondsToSelector:@selector(setPreservesSuperviewLayoutMargins:)]) {
    cell.preservesSuperviewLayoutMargins = NO;
    }
    if ([cell respondsToSelector:@selector(setLayoutMargins:)]) {
    [cell setLayoutMargins:UIEdgeInsetsZero];
    }
    兼容7.0以后的需要设置着三句哈, 没有设置任何一种,在9.0不起作用
  • 静守幸福:关键是cell与cell之间的分割线 我们要求最后一行的cell不能有这样的线
    苹果API搬运工:@静守幸福 弄个[[UIView alloc] init],不要尺寸,添加到tableView的footer试一试
  • YotrolZ:最后写的,第2步和第3部、咋是一样的
    苹果API搬运工:@YotrolZ 写错了,已更正,谢谢 :smile:
  • 267e2f0bbada:很用心
  • 0a87cf525a60:可以少写一个控件了,谢谢
  • 解忧杂货店老板:写得非常不错
  • redSnake:有心了,赞
  • yiron:收了
  • 3eee4f659e67:用的上☺☺
  • loobyes:我一直是在自定义Cell,然后画一个1高度的UIView。设置背景颜色什么的,现在一看,觉得自己好蠢。
    系统盘:@loobyes 能这样想也不错
  • TY_:想请教一下,这种方法设置分割线方法和在布局里设置分割线有什么区别呢
    苹果API搬运工:@TY_ 第三种方法的话系统自带的cell也能用,可以不用自定义了,而且是用系统的属性实现的.其实大部分情况下各种方法效果都差不多,我只是想探究下到底是什么原因造成iOS8后分隔线不能全屏的
  • Callmewenxi:关于第三种方法,直接在- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    这个方法中设置整两个属性就可以了
    cell.layoutMargins = UIEdgeInsetsZero;
    cell.separatorInset = UIEdgeInsetsZero;

    或者自定义一个cell,来设置这两个属性,效果都一样
    薛定谔的熊:@史前图腾 不知道你怎么测试的,反正我刚在8、9、10、11上都试了,只用写这两句即可。
    苹果API搬运工:@小小达文西 我测试的结果是tableview的两个属性也必须要设置,而cell.separatorInset = UIEdgeInsetsZero不写也行.
    另外我觉得该方法最大好处之一是不影响自定义cell,够用就可以不用自定义了,可以偷懒 :sunglasses:

本文标题:UITableView设置全屏分隔线的几种方法比较

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