iOS开发-自定义控件的方式及注意

使用纯代码的方式

  1. 通常来讲咱们的自定义类继承自UIView,首先在initWithFrame:方法中将须要的子控件加入view中。注意,这里只是加入到view中,并无设置各个子控件的尺寸。css

    为何要在initWithFrame:方法而不是在init方法?数组

    由于使用纯代码的方式建立自定义类,在之后使用的时候可能使用init方法建立,也有可能使用initWithFrame:方法建立,可是不管哪一种方式,最后都会调用到initWithFrame:方法。在这个方法中建立子控件,能够保证不管哪一种方式均可以成功建立。ruby

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

    前面已经说过,两种方式最后都会调用到initWithFrame:方法。若是使用init方法建立,那么这个view的frame有多是不肯定的:atom

    CYLView *view = [[CYLView alloc] init];
    view.frame = CGRectMake(0, 0, 100, 100); ...

    若是是这种状况,那么在init方法中,frame是不肯定的,此时若是在initWithFrame:方法中设置尺寸,那么各个子控件的尺寸都会是0,由于这个view的frame尚未设置。(能够看到是在发送完init消息才设置的)spa

    因此咱们应该保证view的frame设置完才会设置它的子控件的尺寸。设计

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

    因此正常的作法应该是在initWithFrame:方法中建立子控件,注意此时子控件有可能只是一个局部变量,因此想要在layoutSubviews访问到的话,通常须要建立这个子控件的对应属性来指向它。regexp

    @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中访问子控件,设置子控件的尺寸,由于此时view的frame已经肯定。对象

    - (void)layoutSubviews { [super layoutSubviews]; // 注意,必定不要忘记调用父类的layoutSubviews方法! self.button.frame = ... // 设置button的frame self.label.frame = ... // 设置label的frame }

    通过以上的步骤,就能够实现自定义控件。

  3. 同时,咱们还但愿能够给咱们的自定义控件数据,让其显示。

    通常来讲首先要将获得的数据转换成模型数据,而后给这个自定义控件传入模型数据让其显示。

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

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

    ...
    @property (nonatomic, strong) Book *book; ...

    在这里咱们接收一个book做为须要显示的数据。

    而后在自定义的实现文件中重写book的setter方法:

    - (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方法给各个控件设置数据

    这样一来就实现自定义类显示数据的功能。并且将子控件封装到自定义中,控制器只须要建立自定义类和给它数据,而不须要担忧这个类内部是怎么设计的,都有什么控件,数据是如何安排的,因此当需求改变时,咱们的控制器有可能彻底不用改动,只需改变自定义类的内部就能够。

总结:

  1. initWithFrame:中添加子控件。

  2. layoutSubviews中设置子控件frame。

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

  4. 在view controller里面使用init/initWithFrame:方法建立自定义类,而且给自定义类的frame赋值。

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

使用xib方式

  1. 使用xib的方式能够省去initWithFrame:layoutSubviews中添加子控件和设置子控件尺寸的步骤,还有在view controller里面设置view的frame,由于添加子控件和设置子控件的尺寸以及整个view的尺寸在xib中就已经完成。(注意整个view的位置尚未设置,须要在控制器里面设置。)

  2. 咱们只需对外提供数据接口,重写setter方法就能够显示数据。

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

  4. 固然,用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];
  5. xib文件中的控件能够经过Control-Drag的方式在CYLView中进行连线,这样CYLView是就能够访问这些控件。(能够在setter方法中给这些控件赋值以显示数据)

总结:

  1. 建立xib,在xib中拖入须要添加的控件并设置好尺寸。而且要将这个xib的Class设置为咱们的自定义类。

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

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

  4. 在view controller类里面加载xib文件就能够获得对应的类(这里不须要再设置自定义类的frame,由于xib已经有了整个view的大小。只须要设置位置。),接着就能够对类对外的数据接口赋值。

补充

  1. 若是使用代码的方式建立控件,那么在建立时必定会调用initWithFrame:方法;若是使用xib/storyboard方式建立控件,那么在建立时必定会调用initWithCoder:方法。

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

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

  3. 事实上使用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才知道。

  4. 若是咱们想,不管是经过代码的方式,仍是经过xib的方式,都会初始化一些值,那么咱们能够将初始化的代码抽到一个方法里面,而后在initWithFrame:方法和awakeFromNib方法中分别调用这个方法。

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

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

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



文/ForeverYoung21(简书做者) 原文连接:http://www.jianshu.com/p/7e47da62899c 著做权归做者全部,转载请联系做者得到受权,并标注“简书做者”。
相关文章
相关标签/搜索