/*
程序启动过程:
main -> UIApplicationMain
***********UIApplicationMain底层做的事情***********
UIApplicationMain底层做的事情:
1.创建UIApplication对象
2.创建AppDelegate对象,并且赋值给UIApplication对象的代理属性
3.开启主运行循环
4.加载info.plist文件,判断是否指定main.storyboard,如果指定,就去加载。
*****************main.storyboard底层做的事情*****************
加载main.storyboard底层做什么事情
1.创建UIWindow对象
2.加载storyboard,创建storyboard中的控制器
3.把新创建的控制器作为窗口的根控制器,让窗口显示在屏幕上
注意:窗口要显示出来,必须保证设置尺寸,避免销毁
*****************窗口补充*****************
常见的窗口:键盘,UIActionSheet,UIAlertView,都会交给application.windows
特殊的窗口:状态栏
窗口的层级:UIWindowLevelNormal
窗口默认都是UIWindowLevelNormal层级
键盘的层级永远是最高的,在显示之前会获取当前最高层级数,然后键盘的层级+1.
一个应用程序只能有一个主窗口
*****************窗口中底层实现方法*****************
rootViewController好处:1.让窗口旋转2.让代码结构清晰(自定义跟控制器,控制器的事情写在控制器里面)
rootViewController底层实现:第一步:让窗口可以旋转
makeKeyAndVisible底层实现:
1.成为application的主窗口 application.keyWindow = self.window;
2.让窗口显示出来 self.window.hidden = NO;
3.把窗口根控制器的view添加到窗口上 [self.window addSubview:window.rootViewController.view];,
self.window = window 底层实现:就会把当前的窗口添加到屏幕上
************通过storyboard创建控制器************
1.加载storyboard
2.实例化控制器(两种方式:1.加载箭头指向的控制器 2.根据标识符创建控制器)
************通过xib创建控制器*****************
1.创建xib的文件
2.往xib里面拖一个view,用来描述控制器的view
3.告诉这个xib是用来描述控制器,设置文件拥有者(file'owner),就可以连线,接着直接连线
错误:
loaded the "VC" nib but the view outlet was not set
解决:
加载xib需要连线,怎么解决,首先你需要告这个xib是描述哪个控制器,设置xib的file'owner为控制器Class
************控制器view的加载(loadView)*****************
loadView作用:创建控制器的view
loadView什么时候去调用:第一次使用控制器的view的时候就会调用loadView
只要重写了loadView这个方法,就必须自己创建控制器的view,系统就不会帮你创建控制器view
HMViewController -> HMView.xib -> HMViewController.xib
loadView底层做法
1.判断下有没有指定storyboard,如果有,就去加载storyboard描述的控制器的view
2.判断下有没有指定nibName,如果有,就去加载nibName描述的控制器的view
2.如何判断有没有指定nibName,[self nibName]
2.1判断下nibName是否为空,如果为空,他会尝试找下有没有跟控制器同名,但是不带Controller的xib
2.2跟控制器同名的xib,nibName = ViewController,但是这一步有条件,前提条件你没有重写loadView
2.3如果都没有找到,直接创建几乎透明的view
init方法底层调用initWithNibName:bundle:
**************************导航控制器基本概念***********************
1.创建导航控制器必须要有一个导航控制器的根控制器,因为导航条的内容必须要依赖栈顶控制器,默认第一个根控制器就是栈顶控制器
2. initWithRootViewController:底层会调用pushViewController。
pushViewController:底层会把控制器压入栈,并且把控制器的view添加到导航条上面。
3.等需要push控制器的view完全显示的时候,会拿到push控制的navigationController属性赋值导航控制器
vc.navigationController = nav;
4.导航控制器的viewControllers用来保存导航控制器里面所有的子控制器
5.导航控制器永远显示出来的界面是栈顶控制器的view
6.调用这个方法,并不会马上出栈,等控制器的view完全移除父控件,才会出栈,控制器出栈后,就会被销毁
********************************导航条的内容***********************
1.导航条的内容由栈顶控制器的UINavigationItem决定,为什么要这样设计,因为导航控制器只有一个导航条,如果每个子控制器都能设置,就不知道听谁的了,所以由显示出来的控制器决定。
2.UINavigationItem:是一个模型,用来决定导航条的内容。
3.UIBarButtonItem:是一个模型,用来决定导航条上按钮的内容。
4.导航条上面的子控件的位置由系统决定,但是尺寸是由我们自己决定
4.1 [button sizeToFit]默认计算按钮尺寸,根据按钮的内容计算出最合适的尺寸
5.在ios7之后,默认就会导航条上的按钮的图片渲染成蓝色
5.1 如何保持图片最原始的效果,不要渲染
[image imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];
5.2Item就是模型,MVC,修改模型就能改界面