在UISplitViewController中,无法生成showDetailViewController:sender:push into detail navigationController

时间:2014-08-03 16:10:15

标签: ios objective-c uinavigationcontroller uisplitviewcontroller ios8

在iOS 8中,视图控制器现在可以调用showDetailViewController:sender:让系统确定正确的视图控制器来呈现详细视图控制器。

在我的应用程序中,我有一个UISplitViewController,它的viewControllers数组中包含两个UINavigationControllers。第一个UINavigationController包含我的“主”视图,UITableViewController的子类。第二个UINavigationController包含我的“详细信息”视图。

由于我正在尝试普遍使用此功能,因此我尝试使用showDetailViewController:sender:来显示详细信息视图:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{

    self.itemVC.item = self.itemStore.items[indexPath.row];

    [self showDetailViewController:self.itemVC sender:self];
}

self.splitViewController.collapsed == YES时,这适用于水平紧凑特征(iPhone风格),但当特征是常规(iPad,未折叠)时则不行。在iPad上,它用细节视图控制器替换细节UINavigationController(而不是替换UINavigationController的viewControllers数组)。

为了解决这个问题,我已经测试了它是否已经崩溃,如果不是,我会在显示它之前将详细视图控制器包装在另一个UINavigationController中:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{

    self.itemVC.item = self.itemStore.items[indexPath.row];

    UIViewController *vcToShow;

    // For whatever reason, when not collapsed, showDetailViewController replaces the detail view, doesn't push onto it.
    if (self.splitViewController.collapsed) {
        vcToShow = self.itemVC;
    } else {
        vcToShow = [[UINavigationController alloc] initWithRootViewController:self.itemVC];
    }

    [self showDetailViewController:vcToShow sender:self];
}

我想我可以配置self.itemVC并避免在showDetailViewController:sender:时完全致电self.splitViewController.collapsed == NO

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{

    self.itemVC.item = self.itemStore.items[indexPath.row];

    // For whatever reason, when not collapsed, showDetailViewController replaces the detail view, doesn't push onto it.
    if (self.splitViewController.collapsed) {
        [self showDetailViewController:vcToShow sender:self];
    }
}

但是,这感觉它正在挫败showDetailViewController:sender:的目的,即放松self与视图层次结构其余部分之间的耦合。

有没有更好的方法来解决这个问题?

3 个答案:

答案 0 :(得分:7)

showDetailViewController:sender:中,根据collapse属性,您需要创建要在详细信息中显示的控制器。

E.g。在横向模式的iPad上,它已经从故事板创建了详细视图控制器,但在iPhone 5上折叠时,视图控制器尚不存在。

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
    UINavigationController *detail;
    ImageViewController *imageVC;

   // on the iPhone (compact) the split view controller is collapsed
   // therefore we need to create the navigation controller and its image view controllerfirst
   if (self.splitViewController.collapsed) {
       detail = [[UINavigationController alloc] init];
       imageVC = [self.storyboard instantiateViewControllerWithIdentifier:@"ImageViewController"];
       [detail setViewControllers:@[imageVC] animated: NO];
   }
   // if the split view controller shows the detail view already there is no need to create the controllers
   else {
       id vc = self.splitViewController.viewControllers[1];
       if ([vc isKindOfClass:[UINavigationController class]]) {
           detail = (UINavigationController *)vc;
           imageVC = [detail.viewControllers firstObject];
       }
    }

    [self prepareImageViewController:imageVC forPhoto:self.photos[indexPath.row]];
    // ask the split view controller to show the detail view
    // the controller knows on iPhone and iPad how to show the detail
    [self.splitViewController showDetailViewController:detail sender:self];
}

我希望这能解决你的问题。

答案 1 :(得分:0)

你这样做的方式有问题。如果在选择后旋转设备(将模式从折叠更改为allVisible),您将找到没有导航控制器的详细信息vc。

如果你在所有情况下调用showDetailViewController:sender:并使用导航控制器传递视图控制器,它将在两种情况下正常工作,并且还将解决上面提到的旋转问题。

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    self.itemVC.item = self.itemStore.items[indexPath.row];

    UIViewController *vcToShow= [[UINavigationController alloc] initWithRootViewController:self.itemVC];
    [self showDetailViewController:vcToShow sender:self];
}

答案 2 :(得分:-1)

if (self.splitViewController.collapsed)
    [self.splitViewController showDetailViewController:self.itemVC sender:self];
else
    self.splitViewController.preferredDisplayMode = UISplitViewControllerDisplayModePrimaryHidden;