美文网首页iOS高质量博客iosiOS
Xib使用之TableViewCell.xib中创建多个Cell

Xib使用之TableViewCell.xib中创建多个Cell

作者: loongod | 来源:发表于2016-04-03 16:34 被阅读12626次

    初次使用xib创建UITableviewCell的时候,我都是一个xib文件里,只创建一个Cell,在实际业务中,往往都是一个列表中需要用到多个不同的Cell样式,这就需要创建N个.h .m .xib文件。而且这些.m中的实现还差不多。
    后来发现,一个.xib文件中可以创建多个Cell,如图:

    多个Cell

    这样感觉方便多了。


    具体实现:

    第一步创建

    先和普通创建xibCell一样,在xib中选中左边那个Cell,copy(command + c)或者在拖一个UITableViewCell上去,然后paste(command + v),.xib中就多个Cell了,O(∩_∩)O~~

    多个Cell
    第二步设置Identifier和代码使用

    在代码中创建Cell

       if (!cell) {
            cell = [[[NSBundle mainBundle] loadNibNamed:@"TempTableViewCell" owner:self options:nil] firstObject];
        }
    

    TempTableViewCell是你的xib文件名,firstObject是第一个Cell,按顺序排的。
    第二个怎么办??

            cell = [[[NSBundle mainBundle] loadNibNamed:@"TempTableViewCell" owner:self options:nil] objectAtIndex:2];
    

    再多依次类推哈。(提示:如果在Cell中添加手势的话,loadNibNamed:这个返回的数组中会比Cell多哦,大家注意)

    设置每个Cell的identifier,(identifier 随意起的,我的规律就是类名+第几,不要重复就行)如图:


    设置每个Cell的identifier

    这样在重用队列中重复使用Cell的时候,能找到正确的Cell,

     TempTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"TempTableViewCellFirst"];
    

    可以根据indexPath设置不同的identifier。
    可以把创建Cell的过程放在Cell.m中,做成类方法,这样不至于VC中的代码过多。
    cell.h中:

    @interface TempTableViewCell : UITableViewCell
    
    /**
     *  @author god~long, 16-04-03 15:04:19
     *
     *  初始化Cell的方法
     *
     *  @param tableView 对应的TableView
     *  @param indexPath 对应的indexPath
     *
     *  @return TempTableViewCell
     */
    + (instancetype)tempTableViewCellWith:(UITableView *)tableView
                                indexPath:(NSIndexPath *)indexPath;
    
    
    @end
    

    cell.m中:

    + (instancetype)tempTableViewCellWith:(UITableView *)tableView indexPath:(NSIndexPath *)indexPath {
        NSString *identifier = @"";//对应xib中设置的identifier
        NSInteger index = 0; //xib中第几个Cell
        switch (indexPath.row) {
            case 0:
                identifier = @"TempTableViewCellFirst";
                index = 0;
                break;
            case 1:
                identifier = @"TempTableViewCellSecond";
                index = 1;
                break;
            case 2:
                identifier = @"TempTableViewCellThird";
                index = 2;
                break;
    
            default:
                break;
        }
        TempTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
        if (!cell) {
            cell = [[[NSBundle mainBundle] loadNibNamed:@"TempTableViewCell" owner:self options:nil] objectAtIndex:index];
        }
        return cell;
    
    }
    

    这样VC中:

    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
        TempTableViewCell *cell = [TempTableViewCell tempTableViewCellWith:tableView indexPath:indexPath];
        return cell;
    }
    

    是不是很方便呢。

    设置属性

    快捷设置xib属性:

    1. 在.h或者.m中先写好属性,如图:
    设置属性
    2. 在xib中就可以拖拽连接属性了,如图:
    property.gif

    重点:关联属性的时候,你想关联那个Cell上的属性,需要先点击左边Cell列表,选中该Cell,然后再拖线关联上面的控件。
    设置好属性,下面就是使用了,
    配置Cell

    /**
     *  @author god~long, 16-04-03 16:04:04
     *
     *  配置TempCell的方法
     *
     *  @param indexPath 对应TableView的indexPath
     */
    - (void)configTempCellWith:(NSIndexPath *)indexPath;
    

    .m中:

    - (void)configTempCellWith:(NSIndexPath *)indexPath {
        switch (indexPath.row) {
            case 0: {
                _label1.text = @"我是Label1";
                _customImageView.image = [UIImage imageNamed:@"8"];
                break;
            }
            case 1: {
                _label2.text = @"我是Label2";
                [_customButton setTitle:@"我是button" forState:UIControlStateNormal];
                break;
            }
            case 2: {
                _label1.text = @"我是第三个Cell的Label1";
                _label2.text = @"我是第三个Cell的Label2";
                break;
            }
            default:
                break;
        }
    }
    

    重点:每个Cell设置链接的属性都是单独处理的,没连,在用的时候即使你用了,也设置不了。
    运行效果:

    运行效果
    深入

    新建一个UITableViewCell创建的时候选择同时创建xib,然后
    右击cell.xib文件➡Open As➡Souce Code 如下图

    open source code

    然后可以看到xib文件其实是以xml数据格式存储的,如图:

    xml

    当我改变xib中1和2两个cell的顺序之后(可以双指拖动),然后查看他们的source code会发现,他们在xml文件中的顺序也会改变。用Bundle.main.loadNibNamed("*xibName*", owner: self, options: nil)方法返回数组的Item顺序也会改变。

    xib中顺序

    我们在调用let cells = Bundle.main.loadNibNamed("*xibName*", owner: self, options: nil)
    的时候,加载的是xml<objects/>标签下的数据,其中<placeholder>标签下中如果placeholderIdentifier="IBFilesOwner"或者"IBFirstResponder"时会被忽略。也就是不会返回到上面的cells数组中。但是如果我添加一个手势的话,就会返回到数组中了。
    具体如下图:

    xib with tapGesture xib with tap souce code cells

    其中<objects>中的默认数据顺序是根据我们在xib中添加控件的顺序而定的,下面我又先添加了一个tap手势,然后又添加了一个cell,查看source code和打印cells数据如下图:

    中间添加手势 cells log
    回答关于重用

    我在创建cell的判断里面

        if (!cell) {
            cell = [[[NSBundle mainBundle] loadNibNamed:@"TempTableViewCell" owner:self options:nil] objectAtIndex:index];
            NSLog(@"create cell indexPath: %ld",index);
        }
    

    然后

    - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
        return 100;
    }
    

    运行并滑动之后截图如下:


    界面 控制台

    So,是会重用滴。

    dome地址:点我点我


    相关文章

      网友评论

      • 90后的晨仔:找到问题了,我在请教个问题。假如用这样的方式自定义三个cell,每个cell上边都有一个UITextField,我要使用UITextFieldDelegate的代理方法怎么指定代理?
      • 90后的晨仔:为什么在控制器中一调用- (void)configTempCellWith:(NSIndexPath *)indexPath,这个方法就蹦呢?
        loongod:控制台打印的崩溃原因呢?仔细检查下呗
      • 5686e30065fc:按照你这个流程走了一遍代码,cell并没有得到复用
        5686e30065fc:@loongod 抱歉抱歉,那天是我测试数量太少了以为没有重用:joy: ,后面自己发现了,要感谢楼主提供的教程~
        loongod:看下我的是怎么重用的:wink: 已在最后面更新。
      • LyTsai:不错不错,我自己也想到了用多个,但是那个连接部分我第一次写错了然后有不少控件nil了,看了楼主的解决了。
      • ZackLi:楼主好像你的 我是 Button 添加不了响应方法?为什么?
        loongod:@李泽昆Fantast 不懂你说的?是关联了方法,没响应吗?还是关联不上?
      • annabelle9899:小白一个, 请问一下看到你的代码, 我想知道你的tableview是在哪里创建的啊? 没有看到任何tableview的创建啊?
      • hext123:这样做的话不会重用了把? 每次都会创建新的应该
        loongod:亲,会重用滴
      • dadahua:学习了
      • Wayne_Wang:嗯,最近正在研究这个东西 但是当我写了个demo 实现的时候发现 重用的时候能正常 走方法 但是在屏幕上显示不出重用的cell 很奇怪的是 这些重用的 还占用了 屏幕的位置(屏幕第一行cell正常第二个开始 到倒数第二个为空的 但是屏幕给出了白色的位置) 就是说 我6个cell 第一个用 第一种自定义 其他5个 用第二种 cell 当第二种cell 重用的时候 就出现了重用占屏幕但是不显示cell 的问题 并且除了 创建的cell 能点击外其他的都不能点击 , 然后我 用[_tableView registerClass:[TableViewCell class] forCellReuseIdentifier:@"tableViewCell2"]; 注册 然后在重用的时候加上
        cell2 = [tableView dequeueReusableCellWithIdentifier:@"tableViewCell2"]; 之后第二种cell方法照样走 但是重用的cell内容全部为空(此时屏幕cell显示正常但是第二种cell 重用的部分内容全部为空) 如果不注册 直接在重用里用cell2 = [tableView dequeueReusableCellWithIdentifier:@"tableViewCell2"]; 就会崩溃。希望能给菜鸟解惑 感激不尽。。。
        loongod:@冷碎音阶丿寒 不要使用registerClass注册cell。手动创建cell可以。
      • 清溪丷:很不错!
        loongod:@清溪丷 嗯,复制一个Cell后,往下拉下就好了。最好和左边排序对应。
        清溪丷:@god_long 不过cell的位置好像有点重叠,他需要在设置下每个cell自己的位置么
        loongod:@清溪丷 谢谢。:yum:
      • 来宝:如果遇到cell重叠的情况怎么办?
        loongod:@来宝 这个应该是cell重用问题吧。
        来宝:@god_long 怎么拖?我遇到的情况是在xib上拖上控件后,然后在其中的某一行用代码加上一个按钮,结果上下滚动的时候这个按钮会重叠
        loongod:@来宝 拖一下就走了。:stuck_out_tongue_winking_eye:
      • angelababa:cell不需要registernib吗??
        loongod:@angelababa 注册的话你可以尝试一下,我没试tableview的。:smile:
        angelababa:@god_long 那这种一个xib里面。多个tableviewcell。registernib要怎么写呢。
        loongod:@angelababa Tableview在iOS7之后才能用注册。
      • 思念那年慕云:你好,我按照这种写法,当多个section的时候,我修改不了cell中的label的内容,我写好了一个属性,然后拖到xib上的
        loongod:@思念那年慕云 好蹦,应该是你的代码有问题,这个你仔细检查下你的代码,再参考下我的Demo,看看那里出错了。
        思念那年慕云:@god_long 我打断点左侧显示的indexPath传到cell中第一行的时候是有值的,后面的每一行断点展示indexPach为nil,但是断点走的一步一步的却是正常的走的,还有我打断点的时候,好崩怎么回事?
        loongod:@思念那年慕云 1,看创建的时候是不是创建的你拖到的那个Cell上,
        2,查看下属性是否关联,
        3,赋值那里断点一下,看看这个属性是否为nil,是nil就说明这个属性没在这个Cell上,
        4,页面创建后,查看下布局层级,看看每个Cell到底有什么控件。
      • Sanchain:已实现collectionView:http://www.codes51.com/itwd/1146028.html
        loongod:@Sanchain :+1: 闲了我也试试。
        Sanchain:@god_long 确定,试过了
        loongod:@Sanchain 这个问题就是我问的,现在还没有解决。https://segmentfault.com/q/1010000004894555。
        你确定可以??
      • Sanchain:请问楼主试过collectionView吗?我刚试了下会崩
      • Sanchain:学习了,谢谢楼主分享。
        loongod:@Sanchain :grin:
      • LeeCen:有没有demo?
        loongod:@Lee_Cen 已上传 https://github.com/god-long/longDemoCollectionmoCollection :relieved:

      本文标题:Xib使用之TableViewCell.xib中创建多个Cell

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