[TOC] 自定义视图

##自定义label-textField视图
|height=2|height=2|width=5|width = 10 |width = 5|
|---|---:|--:|---:|---:|
|width=5|UILabel||UITextField||
|height=2|||||
####.h文件

#import <UIKit/UIKit.h>
@interface LTView : UIView<UITextFieldDelegate>//加了协议
@property (nonatomic,retain) UILabel * laber;
@property (nonatomic,retain) UITextField *textField;
- (instancetype)initWithFrame:(CGRect)frame labelText:(NSString *)labelText placeHolder:(NSString *)placeHolder;
@end

####.m文件

#import "LTView.h"
@implementation LTView
//重写父类的初始化方法,先调用父类的初始化方法,然后在此基础上添加自己的东西
- (instancetype)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self)
    {
        CGFloat height = frame.size.height; //LTView的高度
        CGFloat width = frame.size.width;//LTView的宽度
        CGFloat labelHeight = height - 2 * 2;//label的高度
        CGFloat labelWidth = (width - 5 * 3)/3;//label的高度
        CGFloat textFiledHeight = labelHeight;//TextField的高度
        CGFloat textFieldWidth = 2 * labelWidth;
        
        //创建内部的label
        _laber = [[UILabel alloc] initWithFrame:CGRectMake(5, 2, labelWidth, labelHeight)];
        [self addSubview:_laber];        
        //创建内部的输入框
        _textField = [[UITextField alloc] initWithFrame:CGRectMake(10+labelWidth, 2, textFieldWidth, textFiledHeight)];
        _textField.delegate = self;
        

        _textField.borderStyle = UITextBorderStyleRoundedRect;
        [self addSubview:_textField];
        
     
    }
    return self;
}


- (instancetype)initWithFrame:(CGRect)frame labelText:(NSString *)labelText placeHolder:(NSString *)placeHolder
{
    self = [self initWithFrame:frame];
    if (self) {
        _laber.text = labelText;
       _textField.placeholder = placeHolder;
    }
    return self;
}

###AppDelegate.m文件天下如下代码 记得引用头文件#import "LTView.h"

    CGFloat x = 30;
    CGFloat y = 50;
    NSArray *labelTexts = @[@"用户名",@"密码",@"确认密码",@"邮箱",@"电话号码"];
    NSArray *placeholder = @[@"6~16位字符",@"6~16位字符",@"再一次输入密码",@"email",@"手机号码"];
    for (int i= 0; i < 5 ; i++)
    {
        LTView *ltView = [[LTView alloc] initWithFrame:CGRectMake(x, y, 300, 40)labelText:labelTexts[i] placeHolder:placeholder[i] ];
        ltView.tag = 100 + i;
        [ self.window addSubview:ltView];
        [ltView release];
        y += 60;
    }    
    LTView *ltV = (LTView *) [self.window viewWithTag:102];
    ltV.textField.placeholder = @"请再输入一次密码";

#视图控制器 (处理事件 接收代理)
1.UIViewController:视图控制器
2.控制视图显示,响应事件
3.分担AppDelegate的工作
4.实现模块独立,提高复用性

//创建控制器RootViewController.h/.m 及其创建
//创建一个视图类GetPasswordView
//重写父类的方法,这个方法就一个作用,加载自定义的视图
//loadView这个方法在控制器 第一次使用view的时候,即:第一次[控制器 view]的时候执行!!!第二次就不在执行了.也就是说loadview只执行一次.
//修改AppDelegate.m
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    self.window = [[[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]] autorelease];
    // Override point for customization after application launch.
    self.window.backgroundColor = [UIColor whiteColor];
    //添加内容开始 
    //满足了 AppDelegate的愿望,在程序启动时,给self.window指定了一个根视图控制器.
    //创建一个视图控制器
    RootViewController *rootVC = [[RootViewController alloc] init];
    //指定视图控制器
    self.window.rootViewController = rootVC ;
    //内存处理
    [rootVC release];   
    //添加内容结束 
    [self.window makeKeyAndVisible];
    return YES;
}
-(void)loadView
{
    //self.view = 自定义视图;
    //不管你给定多么大得frame,当它加载时就是按照window的大小 创建的
    GetPasswordView *gtView = [[GetPasswordView alloc] initWithFrame:CGRectMake(0, 0, 0, 0)];//也可以写CGRectZero;
    
    //Button的事件拿出去给控制器
    [gtView.getBt addTarget:self action:@selector(getBack) forControlEvents:UIControlEventTouchUpInside];
    
    //输入框中键盘的收回交给控制器去相应
    gtView.emTf.delegate = self;
    self.view = gtView;
    [gtView release];
}
//键盘回收
- (BOOL)textFieldShouldReturn:(UITextField *)textField
{
    [textField resignFirstResponder];
    return YES;
}
//GetPasswordView.h
#import <UIKit/UIKit.h>
@interface GetPasswordView : UIView
@property (nonatomic,retain) UILabel *emLabel;//邮箱的标签
@property (nonatomic,retain) UITextField * emTf;//邮箱输入框
@property (nonatomic,retain) UIButton *getBt;
@property (nonatomic,retain) UIButton *backBt;
@end
//GetPasswordView.m
#import "GetPasswordView.h"
@implementation GetPasswordView
//要重写dealloc方法释放我们在本类中定义的变量
- (void)dealloc
{
    [_emLabel release];
    [_emTf release];
    [super dealloc];
}
//控制器只负责相应时间,改变根视图的大小,不负责改子视图的大小
//子视图的大小,交给我们自定义的view来修改,
//控制器值负责处理逻辑,不负责界面显示
//这个方法在视图的bounds发生变化时执行.只要bounds发生变化,就会执行这个方法
//横竖颠倒的时候需要调整的
- (void)layoutSubviews
{
    //获取屏幕方法
    UIInterfaceOrientation orientation = [UIApplication sharedApplication].statusBarOrientation;
    if (orientation == UIInterfaceOrientationLandscapeLeft ||
        orientation == UIInterfaceOrientationLandscapeRight
        )//判断home键在左边或者右边  来判断屏幕是横着的还是竖着的
    {
        //此时改变子视图的大小
        _getBt.frame = CGRectMake(100, 150, 90, 40);
        //其他控件和button类似
    }
    else
    {
        _getBt.frame = CGRectMake(30, 150, 90, 40);//此句不可省略,因为竖屏后还会恢复横屏
    }
}

-(instancetype)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self) {
        _emLabel = [[UILabel alloc] initWithFrame:CGRectMake(30, 80, 90, 40)];
        //创建邮箱输入框
        _emLabel.text = @"邮箱";
        [self addSubview:_emLabel];
        _emTf = [[UITextField alloc] initWithFrame:CGRectMake(140, 80, 200, 40)];
        _emTf.borderStyle = UITextBorderStyleRoundedRect;
        _emTf.placeholder = @"请输入邮箱";
        [self addSubview:_emTf];
        
        //创建找回按钮
        _getBt = [UIButton buttonWithType:UIButtonTypeSystem];
        _getBt.frame = CGRectMake(30, 150, 90, 40);
        [_getBt setTitle:@"找回" forState:UIControlStateNormal];
        [self addSubview:_getBt];
        //创建返回按钮
        _backBt = [UIButton buttonWithType:UIButtonTypeSystem];
        _backBt.frame = CGRectMake(150, 150, 90, 40);
        [_backBt setTitle:@"返回" forState:UIControlStateNormal];
        [self addSubview:_backBt];
    }
    return self;    
}

###当一个工程里含有多个视图控制器,我们就要创建一个视图控制器控制器来处理

//.h文件中 引入每个视图的视图控制器,并且把页面跳转的方法写到容器里
#import <UIKit/UIKit.h>
@class LoginViewController;
@class RegisterViewController;
@class FindViewController;

@interface RootViewController : UIViewController

@property (nonatomic, retain) LoginViewController *loginVC;
@property (nonatomic, retain) RegisterViewController *registVC;
@property (nonatomic, retain) FindViewController *findVC;

- (void)login;
- (void)enterRegistView;
- (void)enterGetPasswordView;
- (void)getBackToLoginViewFromRegistView;
- (void)getBackToLoginViewFromFindView;
@end

//.m文件中的方法
-(void)dealloc
{
    [_loginVC release];
    [_registVC release];
    [_findVC release];
    [super dealloc];
}

- (void)login
{
    NSLog(@"登录中");
}

- (void)enterRegistView
{
    [_loginVC.view removeFromSuperview];
    
    _registVC = [[RegisterViewController alloc] init];
//    [_registVC view];//第一次使用视图  第一次调用getter
    
    [self addChildViewController:_registVC];
    [self.view addSubview:_registVC.view];
}

- (void)enterGetPasswordView
{
    [_loginVC.view removeFromSuperview];
    
    _findVC = [[FindViewController alloc] init];
    
    [self addChildViewController:_findVC];
    
    [self.view addSubview:_findVC.view];
}

- (void)getBackToLoginViewFromRegistView
{
    [_registVC.view removeFromSuperview];
    [_registVC removeFromParentViewController];
    [self.view addSubview:_loginVC.view];
}

- (void)getBackToLoginViewFromFindView
{
    [_findVC.view removeFromSuperview];
    [_findVC removeFromParentViewController];
    [self.view addSubview:_loginVC.view];
}

- (void)viewDidLoad {
    [super viewDidLoad];
    //默认的视图写在这边    
    _loginVC = [[LoginViewController alloc] init];
    [self.view addSubview: _loginVC.view];
    
    // Do any additional setup after loading the view.
}
//loadView中创建视图的时候的代码
- (void)loadView
{
    LoginView *loginView = [[LoginView alloc] initWithFrame:[UIScreen mainScreen].bounds];//因为不是直接加载window上面的 如果给0相当于没有 然后 视图越界 点击无效 以后可以都写这种方式
}

###视图控制器中写 键盘从第一输入框 跳到第二输入框

-(BOOL)textFieldShouldReturn:(UITextField *)textField
{
    
    UITextField *tf = (UITextField *)[self.view viewWithTag:102];
    //此处为self.view 本控制器为视图控制器 self.view.window也可以
    if (textField.tag == 101)
    {
        [tf becomeFirstResponder];
        [textField resignFirstResponder];
    }
    else
    {
         [tf resignFirstResponder];
    }
    return  YES;
}

//别忘了代理 还有 设置tag值
    loginView.userFiled.delegate = self;
    loginView.userFiled.tag = 101;
    loginView.passFiled.tag = 102;
    loginView.passFiled.delegate = self;

###内存警告的实现方法

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
    
    //如果当前视图加载完毕,并且没有在window上面进行显示
   if ([self isViewLoaded] && self.view.window == nil)
   {
       //释放资源
       //比如 view上展示的数据(NSArray,NSDictionary,图片,文字之类的Model)
       self.view = nil;
   }

}

总结:最后一个其实也不是很方便,后期会有更便捷的方式

 posted on 2015-04-22 13:55  ianhao_cn  阅读(223)  评论(0编辑  收藏  举报