自定义控件的方式

    xiaoxiao2021-03-25  95

    #pragma mark 自定义控件的方法

    #pragma mark 纯代码自定义控件

    使用纯代码的方式

    一般来说我们的自定义类继承自UIView,首先在initWithFrame:方法中将需要的子控件加入view中。注意,这里只是加入到view中,并没有设置各个子控件的尺寸。

    为什么要在initWithFrame:方法而不是在init方法?

    因为使用纯代码的方式创建自定义类,在以后使用的时候可能使用init方法创建,也有可能使用initWithFrame:方法创建,但是无论哪种方式,最后都会调用到initWithFrame:方法。在这个方法中创建子控件,可以保证无论哪种方式都可以成功创建。

    为什么要在initWithFrame:方法里面只是将子控件加到view而不设置尺寸?

    前面已经说过,两种方式最后都会调用到initWithFrame:方法。如果使用init方法创建,那么这个viewframe有可能是不确定的:

    CYLView *view = [[CYLView alloc] init];

    view.frame = CGRectMake(0,0, 100,100);

    ...

    如果是这种情况,那么在init方法中,frame是不确定的,此时如果在initWithFrame:方法中设置尺寸,那么各个子控件的尺寸都会是0,因为这个viewframe还没有设置。(可以看到是在发送完init消息才设置的)

    所以我们应该保证viewframe设置完才会设置它的子控件的尺寸。

    layoutSubviews方法中就可以达到这个目的。第一次view__将要显示__的时候会调用这个方法,之后当view的尺寸(不是位置)改变时,会调用这个方法。

    所以正常的做法应该是在initWithFrame:方法中创建子控件,注意此时子控件有可能只是一个局部变量,所以想要在layoutSubviews访问到的话,一般需要创建这个子控件的对应属性来指向它。

    @property (nonatomic,weak) UIButton *button;// 注意这里使用weak就可以,因为button已经被加入到self.view.subviews这个数组里。

    ...

    - (instancetype)initWithFrame: (CGRect)frame

    {

        if (self = [super initWithFrame: frame]) {

            UIButton *button = ... //创建一个button

            [button setTitle: ...] //设置button的属性

            [self.view addSubview: button];// button加到view中,并不设置尺寸

            self.button = button;//self.button指向这个button保证在layoutSubviews中可以访问

            

            UILabel *label = ... //其他的子控件同理

        }

    }

    这样我们就可以在layoutSubviews中访问子控件,设置子控件的尺寸,因为此时viewframe已经确定。

    - (void)layoutSubviews

    {

        [super layoutSubviews];// 注意,一定不要忘记调用父类的layoutSubviews方法!

        

        self.button.frame = ...// 设置buttonframe

        self.label.frame = ... // 设置labelframe

        }

        经过以上的步骤,就可以实现自定义控件。

        

        同时,我们还希望可以给我们的自定义控件数据,让其显示。

        

        一般来说首先要将得到的数据转换成模型数据,然后给这个自定义控件传入模型数据让其显示。

        

        所以在这个自定义控件的头文件,需要我们设置接口以得到别人传入的数据。比如当前我们有一个Book类,它有一个name属性用于显示名称,有一个like属性用于显示多少人喜欢。现在我们需要将Bookname显示到自定义类的label子控件上,将Booklike显示到自定义类的button子控件上。

        

        首先在自定义类的头文件中:

        

        ...

        @property (nonatomic, strong) Book *book;

        ...

        在这里我们接收一个book作为需要显示的数据。

        

        然后在自定义的实现文件中重写booksetter方法:

        

        - (void)setBook: (Book *)book

        {

            _book = book;// 注意在这个方法中,不写这句也是没有问题的,因为在下面的语句使用的是book而非self.book_book,但是如果在其他的方法中也想要访问book这个属性,那么就需要写上,否则self.book_book会一直是nil(因为出了这个方法的作用域,book就销毁了,如果再想访问需要有其他的引用指向它)。所以建议,要写上这句。

            

            [self.button setTitle: book.like forState...];

            self.label = book.name;

        }

        这样,当我们想要使用自定义类显示数据时:

        

        // 在控制器类的某个方法中:

        Book *book = self.books[index];// 这里指拿到books这个数据中的某个数据用于显示

        CYLView *view = [[CYLView alloc] initWithFrame: ...];

        [self.view addSubview: view];// 将自定义类加到view

        view.book = book; // 设置book的数据,此时会调用setter方法给各个控件设置数据

        这样一来就实现自定义类显示数据的功能。而且将子控件封装到自定义中,控制器只需要创建自定义类和给它数据,而不需要担心这个类内部是怎么设计的,都有什么控件,数据是如何安排的,所以当需求改变时,我们的控制器有可能完全不用改动,只需改变自定义类的内部就可以。

        

        ​

        

        总结:

        

    initWithFrame:中添加子控件。

        

        layoutSubviews中设置子控件frame

        

        对外设置数据接口,重写setter方法给子控件设置显示数据。

        

        view controller里面使用init/initWithFrame:方法创建自定义类,并且给自定义类的frame赋值。

        

        对自定义类对外暴露的数据接口进行赋值即可。

        

        ​

        

        使用xib方式

        

        使用xib的方式可以省去initWithFrame:layoutSubviews中添加子控件和设置子控件尺寸的步骤,还有在view controller里面设置viewframe,因为添加子控件和设置子控件的尺寸以及整个view的尺寸在xib中就已经完成。(注意整个view的位置还没有设置,需要在控制器里面设置。)

        

        我们只需对外提供数据接口,重写setter方法就可以显示数据。

        

        注意要将xib中的类设置为我们的自定义类,这样创建出来的才是自定义类,而不是默认的父类。

    #pragma mark xib自定义控件所以建议在awakeFromNib方法中进行初始化的额外操作

        当然,用xib这种方式是需要加载xib文件的。加载xib文件有两种方法:

        

        // 第一种方法(较为常用)

        CYLView *view = [[[NSBundle mainBundle] loadNibNamed:@"CYLView" owner:nil options:nil] firstObject]; // CYLView代表CYLView.xib,代表CYLView这个类对应的xib文件。这个方法返回的是一个NSArray,我们取第一个Object或最后一个(因为这个数组只有一个CYLView没有其他对象)就是需要加载的CYLView

        

        // 第二种方法

        UINib *nib = [UINib nibWithNibName:@"CYLView" bundle:nil];

        NSArray *objectArray = [nib instantiateWithOwner:nil options:nil];

        CYLView *view = [objectArray firstObject];

        xib文件中的控件可以通过Control-Drag的方式在CYLView中进行连线,这样CYLView是就可以访问这些控件。(可以在setter方法中给这些控件赋值以显示数据)

        

        总结:

        

        创建xib,在xib中拖入需要添加的控件并设置好尺寸。并且要将这个xibClass设置为我们的自定义类。

        

        通过IBOutlet的方式,将xib中的控件与自定义类进行关联。

        

        所以建议在awakeFromNib方法中进行初始化的额外操作。因为awakeFromNib是在初始化完成后调用,所以在这个方法里面访问属性(IBOutlet)就可以保证不为nil

        

        对外设置数据接口,重写setter方法给子控件设置显示数据。

        

        view controller类里面加载xib文件就可以得到对应的类(这里不需要再设置自定义类的frame,因为xib已经有了整个view的大小。只需要设置位置。),接着就可以对类对外的数据接口赋值。

        

        ​

        

        补充

        

        如果使用代码的方式创建控件,那么在创建时一定会调用initWithFrame:方法;如果使用xib/storyboard方式创建控件,那么在创建时一定会调用initWithCoder:方法。

        

        initWithCoder:里面访问属性,比如self.button,会发现它是nil的,因为此时自定义控件正在初始化,self.button可能还未赋值(self.button是一个IBOutletIBOutlet本质上就相当于Xcode找到这个对应的属性,然后UIButton button = … , [self.view addSubview: button]这种操作,而这一切的操作都是相当于在CYLView view = [[CYLView alloc] initWithCoder:nil]方法之后执行的。上面的代码就相当于用代码的方式实现Xcodestoryboard中加载CYLView),所以如果在这个方法中进行初始化操作是可能会失败的。

        

        所以建议在awakeFromNib方法中进行初始化的额外操作。因为awakeFromNib是在初始化完成后调用,所以在这个方法里面访问属性(IBOutlet)就可以保证不为nil

        

        事实上使用xib创建自定义控件,我们可以将加载xib的过程封装到自定义的类中,只对外暴露一个初始化方法,这样外界就不知道内部是如何创建的自定义控件了。

        

        比如在CYLView.h中提供一个类工厂方法:

        

        + (instancetype)viewWithBook: (Book *)book;

        然后在CYLView.m中实现这个方法:

        

        + (instancetype)viewWithBook: (Book *)book

        {

            CYLView *view = [[[NSBundle mainBundle] loadNibNamed: NSStringFromClass(self) owner:nil opetions: nil] firstObject];

            view.book = book;

            return view;

        }

        这样外界只需用viewWithBook:方法传入一个book,就可以创建一个CYLView的对象,而具体是怎么创建的,只有CYLView才知道。

        

        如果我们想,无论是通过代码的方式,还是通过xib的方式,都会初始化一些值,那么我们可以将初始化的代码抽到一个方法里面,然后在initWithFrame:方法和awakeFromNib方法中分别调用这个方法。

        

        关于为什么是awakeFromNib前面已经说了:

        

        通过xib的方式创建的自定义控件,需要设置IBOutlet属性,虽然会调用initWithCoder:方法,但是调用这个的方法的时候IBOutlet属性还未设置好,所以在这个方法中访问属性将会是nil。而在awakeFromNib中,IBOutlet已经初始化完毕,所以在这个方法中初始化不会失败。

        

        如果通过initWithFrame:方法,说明是通过代码创建的自定义控件,它的属性并不是IBOutlet的,所以不存在未完成IBOutlet的属性未初始化完这种情况。所以在initWithFrame:方法中访问一些属性是没有问题的。但是应该注意,如果是通过init方法创建的自定义控件也会调用initWithFrame:方法,但是此时的self.frame是没有被赋值的(在掉用这个方法的时候并没有设置控件的大小),如果这种情况下使用self.frame是没有值的。注意这种情况。

        

    #pragma mark 自定义cell的方法

        可以通过继承UITableViewCell重新自定义cell,可以像下面一样通过代码来自定义cell,但是手写代码总是很浪费时间,

        

        //CustomTableViewCell.h文件

        

        @interface CustomTableViewCell:UITableViewCell

    @property (nonatomic,strong) UIImageView *headImageView;

    @property (nonatomic,strong) UILabel *nameLabel;

    @property (nonatomic,strong) UILabel *textLabel;

    @end

        

        

        

        //CustomTableViewCell.m文件

        

        @implementation CustomTableViewCell

    - (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier

    {

        

        self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];

        

        if (self) {

            

            // Initialization code

            

            _headImageView = [[UIImageView alloc] init];

            

            [self.contentView addSubView:_headImageView];

            

            _nameLabel = [[UILabel alloc] init];

            

            /*省略Label的属性设置*/

            

            [self.contentView addSubView:_nameLabel];

            

            _textLabel = [[UILable alloc] init];

            

            /*省略Label的属性设置*/

            

            [self.contentView addSubView:_textLabel];

            

        }

        

        returnself;

        

    }

    - (void)layoutSubviews

    {

        

        //省略布局代码

        

        self.headImageView.frame = CGRectMake(.....);

        

        self.nameLabel.frame = CGRectMake(.....);

        

        self.textLabel.frame = CGRectMake(....);

        

        [super layoutSubviews];

        

    }

    @end

        

        上面CustomTableViewCell.h/.m文件中,我用了大段的代码来给自定义的cell布局,真的挺麻烦。然后在ViewController中使用的时候是这样的,

     

        

        - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath

        

        {

            

            static NSString *simpleIdentify =@"SimpleIdentify";

            

            CustomTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:simpleIdentify];

            

            if(cell ==nil)

                

            {

                

                cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:simpleIdentify];

                

            }

            

            //_person就是UITableView的数据源,它里面存放的是Person数据模型,

            

            //Person包括头像headStrnamespeechText等属性

            

            Person *person = [_persons objectAtIndex:indexPath.row];

            

            cell.headImage.image = [UIImage imageNamed:person.headStr];

            

            cell.nameLabel.text = person.name;

            

            cell.textLabel.text = person.speechText;

            

            return cell;

            

        }

        

    下面是代码链接

    http://blog.csdn.net/lvxiangan/article/details/50823086

        

    #prama mark 自定义cell的方法区别 dequeueReuseableCellWithIdentifier: forIndexPath:可不注册的必须判断

        自定义UITableViewCellregisterNib: registerClass:的区别)

        

        自定义UITableViewCell大致有两类方法:

        <>使用nib

        1xib中指定cellClass为自定义cell类型(注意不是设置File's Ownerclass

        

        

        

        2、调用 tableView registerNib:forCellReuseIdentifier:方法向数据源注册cell

        复制代码

        [_tableView registerNib:[UINib nibWithNibName:@"xxxxxCell" bundle:nil] forCellReuseIdentifier:kCellIdentify];

        

        

        3、在cellForRowAtIndexPath中使用dequeueReuseableCellWithIdentifier:forIndexPath:获取重用的cell,若无重用的cell,将自动使用所提供的nib文件创建cell并返回(若使用旧式dequeueReuseableCellWithIdentifier:方法需要判断返回是否为空而进行新建)

        复制代码

        xxxxxCell *cell = [tableView dequeueReusableCellWithIdentifier:kCellIdentify forIndexPath:indexPath];

        

        

        4、获取cell时若无可重用cell,将创建新的cell并调用其中的awakeFromNib方法,可通过重写这个方法添加更多页面内容

        

        

        <>不使用nib

        1、重写自定义cellinitWithStyle:withReuseableCellIdentifier:方法进行布局

        复制代码

        - (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier

        {

            self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];

            if (self)

            {

                // cell页面布局

                [self setupView];

            }

            returnself;

        }

        

        

        2、为tableView注册cell,使用registerClass:forCellReuseIdentifier:方法注册(注意是Class

        复制代码

        [_tableView registerClass:[xxxxxCell class] forCellReuseIdentifier:kCellIdentify];

        

        

        3、在cellForRowAtIndexPath中使用dequeueReuseableCellWithIdentifier:forIndexPath:获取重用的cell,若无重用的cell,将自动使用所提供的class类创建cell并返回

        复制代码

        xxxxxCell *cell = [tableView dequeueReusableCellWithIdentifier:kCellIdentify forIndexPath:indexPath];

        

        

        4、获取cell时若无可重用cell,将调用cell中的initWithStyle:withReuseableCellIdentifier:方法创建新的cell

        

        

        另外要注意的:

        1dequeueReuseableCellWithIdentifier:dequeueReuseableCellWithIdentifier:forIndexPath:的区别:

        前者不必向tableView注册cellIdentifier,但需要判断获取的cell是否为nil

        后者则必须向table注册cell,可省略判断获取的cell是否为空,因为无可复用cellruntime将使用注册时提供的资源去新建一个cell并返回

        

        2、自定义cell时,记得将其他内容加到self.contentView上,而不是直接添加到 cell本身上

        

        

        总结:

        1.自定义cell时,使用什么方式注册

        若使用nib,使用 registerNib:注册,dequeue时会调用 cell -(void)awakeFromNib

        不使用nib,使用 registerClass:注册, dequeue时会调用 cell - (id)initWithStyle:withReuseableCellIdentifier:

        2.需不需要注册?

        使用dequeueReuseableCellWithIdentifier:可不注册,但是必须对获取回来的cell进行判断是否为空,若空则手动创建新的cell

        使用dequeueReuseableCellWithIdentifier:forIndexPath:必须注册,但返回的cell可省略空值判断的步骤。

        //原来的代码地址

    http://www.cocoachina.com/bbs/read.php?tid-290160-page-1.html

    但是如果每个cell的高度不一样,显示的内容不一样,就要按照以下方法自定义

    定义cell的步骤(每个cell的 度不 样,每个cell 显 的内容也不 样)

    1. 新建 个继承 UITableViewCell的 类

    2. initWithStyle: 法中进 控件的初始化 1> 将有可能显 的所有 控件都添加到contentView2> 顺便设置 控件的 些属性( 次性的设置:字体、 字颜 、背景

    3. 提供2个模型 1> 个是数据模型( 字数据 + 图 数据) 2> 个是frame模型(数据模型 + 所有 控件的frame + cell的 度)

    4. cell应该提供 个frame模型属性 1> frame模型传递给cell 2> cell根据frame模型给 控件设置frame,根据数据模型给 控件设置数据 3> cell根据数据模型决定显 和隐藏哪些 控件

    5. tableView的代理 法返回cell的 度 

    转载请注明原文地址: https://ju.6miu.com/read-20398.html

    最新回复(0)