斯坦福大学IOS开发课程笔记(第七课第二部分)

转载请注明出处

http://blog.csdn.net/pony_maggie/article/details/32163347

作者:小马

本篇是demo演示,程序其实就是上节课的心理学家那个demo, 不过在这个demo的基础上,把它作成一个通用版,可以自动识别当前运行的设备是iphone还是ipad,然后有不同的显示效果。所以,还是打开原来的Psychologist工程,在开始之前,我会用第五课讲的自动布局知识,让视图能在横屏模式下也可以正常显示。这一部分不细讲,参考第五课。

下面就可以讲如何能做一个通用的app来同时适配iphone和ipad。 我们希望在iphone上的运行效果是这样的(详细的可以参考第五课的博文):

 

而ipad上,就会用到分屏视图,显示效果如下图所示:

  

好,开始了。首先,改一下工程配置,如下图:

可以马上运行看看效果,选择ipad模拟器,然后run,效果如下所示:

 

屏幕与视图的对比效果反差太大,就像把iphone上的显示直接copy到ipad上并强制拉伸了一样,非常丑。这是因为我们在ipad运行的还是原来iphone上的storyboard。Ipad需要有自己的storyboard。新建一个storyboard方法如下:

第一步,重命名原来的storyboard为Iphone.storyboard(名字随便起,只是为了区分)。

第二步,增加一个新的storyboard,起名为Ipad.storyboard。步骤如下图:

 

第三步,在工程设置里指定 main interface为刚才新建的这个storyboard,如下

ipad.storyboard是空的,首先给他增加一个分屏视图控制器。如下图:

把table view controller删除,这个用不到,然后右边剩下的两个,窄一点的是左边栏(master), 宽的是右边栏(detail),还记得上一部分讲的这两个概念吧。根据文章一开始给出的效果图来看,右边栏应该是要显示”笑脸”的, 所以要把这一部分功能重用,具体方法不说了,在第六课的博文中,我详细的讲过重用的步骤,简单来讲就是把控制对应的类设置成HappinessViewController,增加一个view,
对应的类设置成FaceView,然后连接好outlet。

左边栏的处理相对简单,我们从iphone.storyboard里把导航栏相关的几个controller复制过来,然后粘贴到ipad.storyboard里就要以了,然后拖一些segue 建立push关系,如下图:

接下来,我们建立左边栏和右边栏的连接,是否还记得在iphone.storyboard里,对于Dr.Freud, 我们是先新建一个segue,然后标识到,接着调用performSegueWithIdentifier来实现切换视图效果,那么对于ipad,因为有了分屏视图,可以在一屏显示两个视图,所以我们只要发个消息设置笑脸的幸福数就行了,如下:

//这个函数可以判断当前运行的设备是ipad还是iphone, 因为只有ipad才有split view
- (HappinessViewController *)splitViewHappinessController
{
    id hvc = [self.splitViewController.viewControllers lastObject]; //最后一个是右边栏
    if (![hvc isKindOfClass:[HappinessViewController class]])
    {
        hvc = nil;
    }
    return hvc;
}

- (void)setAndShowDiagnosis:(int)diagnosis
{
    self.diagnosis = diagnosis;
    if ([self splitViewHappinessController])
    {
        [self splitViewHappinessController].happiness = diagnosis;//直接设置
    }
    else
    {
        [self performSegueWithIdentifier:@"ShowDiagnosis" sender:self];
    }

}

注意到代码里的splitViewHappinessController函数,我们用它来判断当前运行的设备是ipad还是iphone(当然只是这个demo这样可以,对于没有splitview的ipad应用,还有其它方法), 因为只有ipad才有分屏视图,那么当前的viewController的splitViewConctroller的viewControllers(有点绕)属性才有分屏对应的两个元素。这里还有一个问题,究竟是什么时候触发的self.splitViewController.viewControllers有左右栏两个元素呢?
应该是当把splitViewCtroller拖进storyboard后,所有在这个集合里的view controller就都具备了。

好了,到这里关于Dr.Freud相关的已经设置好了,可以运行看看效果,篇幅有限,这里不上图了。Dr.Pill相关的因为有三个不同的segue,情形稍复杂一些,不过也不难,篇幅有限,这里不处理了,有兴趣的可以自己做。

刚才都是在横屏状态下调试程序,我还没有看到竖屏的效果,它是这样的:

左边栏没有了,右边栏占据整个屏幕,当然不是我们想要的,正如第一部分讲到的,竖屏状态下我们期望有一个工具栏,然后工具栏有一个按钮可以弹出popover风格的左边栏。

首先我们要给分屏视图控制器指定一个代理(委托), 用哪个控制器作为代理呢? 因为要控制的是左边栏,所以要考虑左边的几个控制器,有三个,那很自然的就会选导航的根视图控制器了,因为两个医生的控制器并不是每次都出现,而根视图控制器是每次都会加载的,所以选它们的上级。代码如下:

//实现协议的方法
//某个状态下是否隐藏左边栏
- (BOOL)splitViewController:(UISplitViewController *)svc shouldHideViewController:(UIViewController *)vc inOrientation:(UIInterfaceOrientation)orientation
{

    return NO;
}

运行看看效果:

恩,效果已经好很多了,不过我们还想再优化一下,因为右边栏感觉被挤压了,试试popover的效果。

我们需要实现协议的另两个方法:

- (void)splitViewController:(UISplitViewController *)svcwillHideViewController:(UIViewController*)aViewController withBarButtonItem:(UIBarButtonItem*)barButtonItem forPopoverController:(UIPopoverController *)pc;

//Called when the view is shown again in the split view, invalidating the buttonand popover controller.
- (void)splitViewController:(UISplitViewController *)svcwillShowViewController:(UIViewController *)aViewControllerinvalidatingBarButtonItem:(UIBarButtonItem*)barButtonItem;

这两个方法分别在左边栏将要隐藏时调用(比如由横屏转到竖屏)和左边栏将要显示时调用(比如由竖屏转到横屏)。我们将在第一个函数的实现显示工具栏上的按钮,第二个函数隐藏工具栏上的按钮。

这里思考一个问题,谁来显示和隐藏工具栏上的按钮呢,答案是右边栏视图控制器,因为在竖屏状态下,只有它是显示的,所以我们应该在这视图下放一个工具栏,然后由右边栏视图控制器来操纵这个按钮。再进一步思考,如何把上面的行为作得更通用,方法是增加一个协议,这个协议可以控制显示或隐藏按钮,然后右边栏视图控制器来实现这个协议。

有了上面的理论基础,就可以开始行动了,首先增加一个协议,如下;

#import <UIKit/UIKit.h> //注意这里,改成UIKit

@protocol SplitViewBarButtonItemPresenter <NSObject>

@property (nonatomic, strong) UIBarButtonItem *splitViewBarButtonItem;

@end

回到PassViewController, 增加一个方法获取右边栏的对象,这样才能操作按钮,方法如下:

//获取右边栏对象
- (id <SplitViewBarButtonItemPresenter>)splitViewBarButtonItemPresenter
{
    id detailVC = [self.splitViewController.viewControllers lastObject];
    if (![detailVC conformsToProtocol:@protocol(SplitViewBarButtonItemPresenter)])
    {
        detailVC = nil;
    }
    return detailVC;
}

协议写好之后,可以来看看上面提到的两个方法的实现了,如下:

//实现协议的方法
//某个状态下是否隐藏左边栏
- (BOOL)splitViewController:(UISplitViewController *)svc shouldHideViewController:(UIViewController *)vc inOrientation:(UIInterfaceOrientation)orientation
{
    //注意理解这一行代码,它有两部分含义,
    //一是如果没有获取到右边栏控制器或该控制器没有实现协议,就直接返回NO,让左边栏永远显示
    //二是如果一不成立,就在竖屏时隐藏左边栏
    return [self splitViewBarButtonItemPresenter]? UIInterfaceOrientationIsPortrait(orientation) : NO;
}

//左边栏将要隐藏时调用(比如由横屏转到竖屏)
- (void)splitViewController:(UISplitViewController *)svc willHideViewController:(UIViewController *)aViewController withBarButtonItem:(UIBarButtonItem *)barButtonItem forPopoverController:(UIPopoverController *)pc
{
    //做两件事
    //一,把工具栏上按钮的title设置成self.title(这里是"doctor")
    //二,把按钮放到工具栏上,这个任备要由右边栏完成,因为这个时候只有右边栏是显示的。

    barButtonItem.title = self.title;
    [self splitViewBarButtonItemPresenter].splitViewBarButtonItem = barButtonItem;

}

//左边栏将要显示时调用(比如由竖屏转到横屏)
- (void)splitViewController:(UISplitViewController *)svc willShowViewController:(UIViewController *)aViewController invalidatingBarButtonItem:(UIBarButtonItem *)barButtonItem
{
    //把工具栏上的按钮移除,因为左边栏显示,所以不需要
    [self splitViewBarButtonItemPresenter].splitViewBarButtonItem = nil;
}

最后,要操作一下右边栏视图控制器,首先在storyboard里,给他增加一个工具栏,然后把默认的按钮删掉(我们要自己控制,所以不用它自带的)并给这个工具栏连接一个outlet以便于操作。接着继承协议并实现,代码如下:

//setter方法
- (void)setSplitViewBarButtonItem:(UIBarButtonItem *)splitViewBarButtonItem
{
    if (_splitViewBarButtonItem != splitViewBarButtonItem)
    {
        NSMutableArray *toolBarItems = [self.toolBar.items mutableCopy];
        if (_splitViewBarButtonItem)
        {
            //删除原来的
            [toolBarItems removeObject:_splitViewBarButtonItem];
        }
        if (splitViewBarButtonItem)
        {
            [toolBarItems insertObject:splitViewBarButtonItem atIndex:0]; //放在最左边
        }
        self.toolBar.items = toolBarItems;
        _splitViewBarButtonItem = splitViewBarButtonItem;
    }
}

好了,运行一下,会看到文章最开始给出的效果。

代码下载地址:

http://download.csdn.net/detail/pony_maggie/7502567

斯坦福大学IOS开发课程笔记(第七课第二部分)

时间: 2024-10-02 01:30:43

斯坦福大学IOS开发课程笔记(第七课第二部分)的相关文章

斯坦福大学IOS开发课程笔记(第七课第一部分)

转载请注明出处 http://blog.csdn.net/pony_maggie/article/details/31462099 作者:小马 这节课的内容太多,分两部分介绍.本节课主要是介绍怎样开发基于ipad的应用程序.涉及到了一些仅仅有ipad才有的类,接口等. 先讲到UIToolbar工具栏.经常使用的UI控件.在工具栏上button一样的东西叫UIBarButtonItems,能够像一般button一样操作它,比方建立outlet,action等.使用的时候仅仅要从工具箱里拖到视图中就

IOS学习之斯坦福大学IOS开发课程笔记(第六课)

转载请注明出处 http://blog.csdn.net/pony_maggie/article/details/28398697 作者:小马 这节课主要讲述多个MVC是如何协同工作的.到目前为止,所有的课程都只是涉及到一个视图.从这节课开始,将会持到多视图应用的例子. 多视图开发这里用navigationController举例子.这个也是比较常用的.如上图所示,一个 navigation Controller(MVC)会指向一个rootViewController(另一个MVC),这个是第一

斯坦福大学IOS开发课程笔记(第八课)

转载请注明出处 http://blog.csdn.net/pony_maggie/article/details/37370159 作者:小马 这节课讲视图的生命周期,网络视图,图像视图以及 滚动视图这些概念以及相关的demo演示.前两个概念比较简单,看一看就明白,我这里只是写图像视图以及 滚动视图的课程笔记. UIImageView用来显示图片.在下面的代码示例中,会有详细的步骤说明,有时候一些简单的应用,可以直接通过xcode设置,甚至不需要写代码. 这幅图向我们展示了scroll view

斯坦福大学 iOS 开发公开课总结

 斯坦福大学 iOS 开发公开课总结 前言 iPhone 开发相关的教程中最有名的,当数斯坦福大学发布的 “iPhone 开发公开课 “ 了.此公开课在以前叫做<iPhone 开发教程>,今年由于平板电脑的流行,所以也加入了 ipad 开发相关的课程.在 网易公开课 上,有 该教程 的 2010 年录象,并且前面 15 集带中文字幕文件,非常适合初学者学习. 在这里顺便说一下,网易公开课上的 28 集其实并不需要全部看完.真正的课程只有前面 12 集.后面的课程都是请一些业界的名人讲他们成功的

斯坦福大学iOS开发作业

今晚把斯坦福大学iOS开发视频的第二课看完了,同时把教授布置的作业也顺手做了下. 教授布置的作业是:在已有的纸牌游戏APP程序基础上拓展,使每次翻牌时出现随机的花色. 其实这个很简单,就是把教授上课写好的类进行实例化,同时获取纸牌的具体内容,即contents属性. 下面我就把做完的APP和APP类结构的图贴上来.          APP运行图:           that's all. Thank you!! 斯坦福大学iOS开发作业

IOS开发学习笔记-(2)键盘控制,键盘类型设置,alert 对话框

一.关闭键盘,放弃第一响应者,处理思路有两种 ① 使用文本框的 Did End on Exit 绑定事件 ② UIControl on Touch 事件 都去操作 sender 的  resignFirstResponder #import <UIKit/UIKit.h> @interface ViewController : UIViewController @property (weak, nonatomic) IBOutlet UITextField *txtUserName; @pro

iOS开发学习笔记:基础篇

iOS开发需要一台Mac电脑.Xcode以及iOS SDK.因为苹果设备都具有自己封闭的环境,所以iOS程序的开发必须在Mac设备上完成(当然,黑苹果应该也是可以的,但就需要花很多的精力去折腾基础环境),Xcode是一个集成开发环境,包括了编辑器.调试.模拟器等等一系列方便开发和部署的工具,iOS SDK则是开发应用所必需,不同的SDK分别对应不同的iOS版本或设备,通常我们需要下载多个iOS SDK以确保我们开发的程序能够在不同版本的iOS上正常运行. 创建新工程 Xcode提供了很多种工程模

IOS开发学习笔记-(3) 进度条、等待动画开始停止

一.创建对应空间视图  ,如下图: 二.编写对应的 .h 代码,如下 : #import <UIKit/UIKit.h> @interface ViewController : UIViewController @property (weak, nonatomic) IBOutlet UIActivityIndicatorView *activWaitNetWork; @property (weak, nonatomic) IBOutlet UIProgressView *pgrsDownLo

IOS开发复习笔记(3)-ARC

1.ARC 当你自己调用了release或retain语句的时候,ARC有效时编译文件会遇到错误,你可以通过-fno-objc-arc和-fobjc-arc两个编译器标志在混搭中支持ARC和非ARC的代码 如下面编译支持ARC,而文件代码不支持ARC # if !__has_feature(objc_arc) //this code do not support to ARC -(void) release{ //release your var } #endif 在ARC工程中集成非ARC的第