0

我是 Objective-C 的新手,所以请耐心等待。我从 Xcode4 中的 Universal App 模板开始构建我的应用程序。有一个约定,模板让您开始使用我试图坚持的约定。对于每个 View Controller,每个设备类型都有一个主文件和一个子类。例如:

Project/
    ExampleViewController.(h|m)
    - iPhone/
      - ExampleViewController_iPhone.(h|m|xib)
    - iPad/
      - ExampleViewController_iPad.(h|m|xib)

在大多数情况下,这非常方便。大多数逻辑都在超类中,子类负责任何特定于设备的实现。

这是我不明白的部分。有时我的代码在每个子类中执行相同的操作只是因为我需要为每个设备加载不同的 xib。例如:

ExampleViewController_iPhone

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    Content *selectedContent = (Content *)[[self fetchedResultsController] objectAtIndexPath:indexPath];
    ContentDetailViewController_iPhone *detailViewController = [[ContentDetailViewController_iPhone alloc] init];
    detailViewController.content = selectedContent;
    detailViewController.managedObjectContext = self.managedObjectContext;

    [self.navigationController pushViewController:detailViewController animated:YES];
    [detailViewController release];
}

ExampleViewController_iPad

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    Content *selectedContent = (Content *)[[self fetchedResultsController] objectAtIndexPath:indexPath];
    ContentDetailViewController_iPad *detailViewController = [[ContentDetailViewController_iPad alloc] init];
    detailViewController.content = selectedContent;
    detailViewController.managedObjectContext = self.managedObjectContext;

    [self.navigationController pushViewController:detailViewController animated:YES];
    [detailViewController release];
}

...请注意,在第二种情况下,唯一不同的是它正在加载_iPadView Controller 的版本。这是必要的,因为iPadiPhone视图控制器附加到单独的设备特定 nib。

这样做的“正确”模式是什么?


更新

我找到了这个关于使用设备修饰符加载单独的 xib 的答案,这似乎在我不需要为一个设备提供特定子类的情况下有所帮助,但如果我需要实例化一个特定的_iPhone_iPad实例,它仍然无济于事用于设备特定功能的视图控制器。

4

1 回答 1

1

有两种简单的方法可以解决您的问题。当放置在您的超类中时,两者都将起作用。

第一种方法之所以有效,是因为您有两个不同的类,而创建哪个类取决于所使用的设备。如果您不使用不同的类,它就不起作用,因为没有特定于设备的代码。它涉及要求对象的类来决定它是哪个类。由于对象的类将是特定于设备的类,即使在超类要求时,您也可以查看创建了哪个类并采取相应措施。

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    Content *selectedContent = (Content *)[[self fetchedResultsController] objectAtIndexPath:indexPath];
    Class classToUse;
    if([self class] == [ExampleViewController_iPad class]) {
        // We are running on the iPad
        classToUse = [ContentDetailViewController_iPad class];
    } else {
        // We must be running on either the iPhone or iPod touch
        classToUse = [ContentDetailViewController_iPhone class];
    }
    // Just use superclass for typing here, since we aren't doing anything device specific
    ContentDetailViewController *detailViewController = [[classToUse alloc] init];
    detailViewController.content = selectedContent;
    detailViewController.managedObjectContext = self.managedObjectContext;

    [self.navigationController pushViewController:detailViewController animated:YES];
    [detailViewController release];
}

第二种方法适用于程序中的任何位置。Apple 在 iOS 3.2 中向 UIDevice 类添加了一个属性,称为“用户界面习语”。目前有两个可能的值:UIUserInterfaceIdiomPadUIUserInterfaceIdiomPhone。由于这些在 3.2 之前的版本中不存在,Apple 还添加了一个宏,UIUserInterfaceIdiomPhone如果版本小于 3.2,它将返回,如果大于或等于 3.2,则从 UIDevice 对象获取实际值。

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    Content *selectedContent = (Content *)[[self fetchedResultsController] objectAtIndexPath:indexPath];
    Class classToUse;
    // If you aren't supporting versions prior to 3.2, you can use [UIDevice currentDevice].userInterfaceIdiom instead to save a couple of cycles
    if(UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad) {
        // We are running on the iPad
        classToUse = [ContentDetailViewController_iPad class];
    } else {
        // We must be running on either the iPhone or iPod touch
        classToUse = [ContentDetailViewController_iPhone class];
    }
    // Just use superclass for typing here, since we aren't doing anything device specific
    ContentDetailViewController *detailViewController = [[classToUse alloc] init];
    detailViewController.content = selectedContent;
    detailViewController.managedObjectContext = self.managedObjectContext;

    [self.navigationController pushViewController:detailViewController animated:YES];
    [detailViewController release];
}
于 2011-04-12T04:45:00.477 回答