7

我有一个名为 vc0 的视图控制器,其显示如下:

[self presentViewController: vc1 animated: YES completion: nil];

在 vc1 中,我有一个按钮来展示另一个视图控制器:

[self presentViewController: vc2 animated: YES completion: nil];

然后在 vc2 中,我有一个关闭视图控制器的按钮:

[self dismissViewControllerAnimated:YES completion: ^{
// over here I call one method in vc1
}

正如预期的那样,它返回到 vc1.. 但是 vc1 中有一个按钮可以通过像这样关闭视图控制器来返回到 vc0:

    [self dismissViewControllerAnimated:YES completion:nil];

但由于某种原因它不起作用,视图控制器不会被解散回 vc0。当我第一次展示 vc1 时,我可以按下按钮来关闭视图控制器并且它可以工作。但是当我按下按钮打开 vc2,当我关闭 vc2 回到 vc1,然后我按下按钮关闭视图控制器时,它就不起作用了。

抱歉,如果问题有点不清楚,我想说的话有点难以表达。

还有一件事:

我尝试dismissViewControllerAnimated:在 vc1 中替换以手动呈现 vc0,但随后我在控制台中收到一条日志,说我正在尝试呈现 vc0 但 vc1 的视图不在窗口层次结构中。这是什么意思?

感谢帮助!

更新:

在这种情况下 VC0 是MenuMileIndexViewController- VC1 是FlightViewController- VC2 是BookmarksTableViewController

这是涉及的代码:

MenuMileIndexViewController:

- (IBAction)goToOriginPage {

FlightRecorder *origin = [[FlightRecorder alloc] init];
[self presentViewController:origin animated:YES completion:nil];

}

飞行记录仪:

    - (void)searchBarBookmarkButtonClicked:(UISearchBar *)searchBar {

        [self bringUpBookmarkkTable];
}

- (void) bringUpBookmarkkTable {

    BookmarkTableViewController *bookmarkTVC = [[BookmarkTableViewController alloc] init];

    [bookmarkTVC setModalTransitionStyle: UIModalTransitionStyleFlipHorizontal];

    [self presentViewController:bookmarkTVC animated:YES completion:nil];
}

- (IBAction)cancel {

[self dismissViewControllerAnimated:YES completion:nil];

}

- (void)endBookmarkProcessWithBookmarkCollection: (NSDictionary *)dict {

    presetBookmarkContext = [dict mutableCopy];

    bookmarkMode = YES;

    NSString *compiledText = nil;

    NSNumber *number1 = [NSNumber numberWithInt: 1];

    if ([dict objectForKey: @"bookmarkTag"] == number1) {

        compiledText = [NSString stringWithFormat: @"%@ to %@", [dict objectForKey: @"origin"], [dict objectForKey: @"destination"]];
    }
    else {

        compiledText = [NSString stringWithFormat: @"%@ to %@", [dict objectForKey: @"destination"], [dict objectForKey: @"origin"]];
    }

    compiledText = [compiledText stringByReplacingOccurrencesOfString:@"Origin: " withString:@""];

    compiledText = [compiledText stringByReplacingOccurrencesOfString:@"Destination: " withString:@""];

    flightContext = [NSDictionary dictionaryWithObjectsAndKeys: [dict objectForKey: @"miles"], @"miles", compiledText, @"location", [[NSUserDefaults standardUserDefaults] objectForKey: @"tempD"], @"date", nil];

    NSString *string = [NSString stringWithFormat: @"\nMiles: %.2f\nFlight: %@\nDate: %@", [[dict objectForKey: @"miles"] floatValue], compiledText, [[NSUserDefaults standardUserDefaults] objectForKey:@"tempD"]];

    UIAlertView *bvkBookmarkAlertView = [[UIAlertView alloc] initWithTitle:@"Confirmation" message:string delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Add", nil];

    [bvkBookmarkAlertView show];
}



 - (void)alertView:(UIAlertView *)alertView didDismissWithButtonIndex:(NSInteger)buttonIndex {

    if (buttonIndex == 1) {

        [self cancel]; // Even though cancel is an IBAction, IBAction is the same thing as void so it is callable
    }
}

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {

    if (buttonIndex == 1) {

        [TheMileIndexViewController addDesiredMilesToIndex: [[flightContext objectForKey: @"miles"] doubleValue]];

        [TravelLogViewController addFlight: flightContext];

        if (!bookmarkMode) {

            if ([checkbox isSelected]) {

                [BookmarkHandler uploadBookmark: bookmarkFlightContext];
            }    
        }
    }

    if (buttonIndex == 0) {

        if ([alertView.title isEqualToString: @"Confirmation"]) {

            bookmarkMode = NO;
        }
    }

}

书签表视图控制器:

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

    [tableView deselectRowAtIndexPath:indexPath animated: YES];

    NSDictionary *dict = [[BookmarkHandler bookmarkCollection] objectAtIndex: indexPath.row];

    fl = [[FlightRecorder alloc] init];

    [self dismissViewControllerAnimated:YES completion:^{

        [fl endBookmarkProcessWithBookmarkCollection: dict];
    }];
}

现在,我在模拟器中创建了应用程序的屏幕录像,显示了问题所在。我可以通过电子邮件将其发送给您以供参考。所以我可以通过电子邮件将其发送给您。

4

3 回答 3

6

我建议始终从实际呈现它的 VC 中解散 VC - 使用委托。这实际上也是 Apple 推荐的方式——正如我在之前对我关于这个问题的问题的回答中指出的那样。

因此,如果您有 VC0 呈现 VC1,那么在 VC0 中也有关闭 VC1 代码,使用委托方案。

我了解到这是处理呈现和解散的最节省的方法——即使有时它可以在 VC1 本身内解散 VC1。

我问了一个非常相关的问题,您可能也有兴趣检查一下。它显示代码...

ps我还读到有些人只解雇VC1 - 这反过来也会解雇VC2。但是,如果我之前的建议有效,我不会这样做。有时我在执行过程中得到信息(没有崩溃),VC 不再存在或与之相关的任何东西 - 所以我认为这不是最好的解决方案。但是如果我之前的建议不起作用,你可以试试第二个。

虽然不能保证这会持续到新 iOS 的更新,但由于这个问题一直困扰着我,现在有几个 iOS 更新:-),所以我决定走标准推荐路线。

编辑:这是我修改后的代码-它可以正常工作-但是尚不清楚在用户响应警报之后您打算发生什么以及警报应该在VC1还是VC0上。无论如何使用委托和回调我没有看到任何问题。如果我错过了您的观点,请解释...

FlightViewControllerProtocol.h

@protocol FlightViewControllerProtocol <NSObject>
-(void) dismissVCAndEndBookmark;
@end

FlightViewController.m

#import "FlightViewController.h"
#import "FlightViewControllerProtocol.h"
#import "BookmarksTableViewController.h"
@interface FlightViewController ()

@end

@implementation FlightViewController
@synthesize delegate;

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view from its nib.
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

- (void)searchBarBookmarkButtonClicked:(UISearchBar *)searchBar {

    [self bringUpBookmarkkTable];
}

- (IBAction) bringUpBookmarkkTable {

    BookmarksTableViewController *bookmarkTVC = [[BookmarksTableViewController alloc] init];
    bookmarkTVC.delegate = self;
    [bookmarkTVC setModalTransitionStyle: UIModalTransitionStyleFlipHorizontal];

    [self presentViewController:bookmarkTVC animated:YES completion:nil];
}

- (IBAction)cancel {

    [self dismissViewControllerAnimated:YES completion:nil];

}

- (void)endBookmarkProcessWithBookmarkCollection: (NSDictionary *)dict {

//    presetBookmarkContext = [dict mutableCopy];

//    bookmarkMode = YES;

    NSString *compiledText = nil;

    NSNumber *number1 = [NSNumber numberWithInt: 1];

    if ([dict objectForKey: @"bookmarkTag"] == number1) {

        compiledText = @"Text1";
    }
    else {
        compiledText = @"Text2";
    }


//    flightContext = [NSDictionary dictionaryWithObjectsAndKeys: [dict objectForKey: @"miles"], @"miles", compiledText, @"location", [[NSUserDefaults standardUserDefaults] objectForKey: @"tempD"], @"date", nil];

    NSString *string = compiledText;

    UIAlertView *bvkBookmarkAlertView = [[UIAlertView alloc] initWithTitle:@"Confirmation" message:string delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Add", nil];

    [bvkBookmarkAlertView show];
}


- (void) dismissVCAndEndBookmark {
    [self dismissViewControllerAnimated:YES completion:nil];
     [self endBookmarkProcessWithBookmarkCollection: nil];
}

- (void)alertView:(UIAlertView *)alertView didDismissWithButtonIndex:(NSInteger)buttonIndex {

    if (buttonIndex == 1) {

        [self cancel]; // Even though cancel is an IBAction, IBAction is the same thing as void so it is callable
    }
}

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {

    if (buttonIndex == 1) {

        NSLog(@"alertView1");
           }

    if (buttonIndex == 0) {
        NSLog(@"alertView2");

    }

}

@end

书签TableViewController.h

 @interface BookmarksTableViewController : UIViewController
{
    id delegate;
}

@property (nonatomic,strong)   id delegate;

@end

书签TableViewController.m

- (IBAction)goBack {
    [self.delegate dismissVCAndEndBookmark];
}

如果我正确理解您的意图,特别是 BookmarksTableViewController.m 中的回调似乎是您实现中的主要问题。

于 2012-09-29T06:35:13.093 回答
6
[self.navigationController popViewControllerAnimated:YES];

也为我做了伎俩。就我而言,我有一个(iPhone)viewController 使用 push segue 被推送到堆栈中。由于启动 segue 的 viewController 有一个导航栏,我不得不向父控制器的 navigationController 发送 popViewControllerAnimated 消息,而不是调用它自己的 dismissViewControllerAnimated:completion 消息。

射线

于 2013-04-03T18:20:37.790 回答
0

第一视角:

首先在第一个视图中嵌入导航控制器。并使用此代码导航到另一个视图

UIStoryboard *storyboard = [UIStoryboard storyboardWithName:@"MainStoryboard" bundle: nil]; 
FirstView *rvc = [storyboard instantiateViewControllerWithIdentifier:@"apps"];
[self.navigationController pushViewController:rvc animated:YES];

第二种观点:

在关闭视图的方法中添加此行

- (IBAction)back{
    [self.navigationController popViewControllerAnimated:YES];
}
于 2012-09-29T07:41:14.293 回答