2

我有一个视图控制器,它查询网络服务是否应该显示插页式广告。如果是这样,则使用 presentViewController:animated:completion: 实例化和呈现另一个视图控制器。根据这个答案文档,我假设 viewDidAppear: 在关闭呈现的视图控制器(它自己做)时不会被调用。从概念上讲,无论如何对我来说,呈现视图控制器的视图永远不会从视图层次结构中删除,因此永远不需要“重新出现”。我显然错了。那么发生了什么?为什么我看到的与文档所说的不同?

- (void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];

    [[AdService sharedAdService] adForSlotName:@"Main Interstitial" completionBlock:^(Ad *adForSlotName) {

        if(adForSlotName)
        {
            InterstitialAdViewController_iPhone *interstitialAdViewController = [[InterstitialAdViewController_iPhone alloc] init];
            interstitialAdViewController.ad = adForSlotName;

            dispatch_queue_t mainQueue = dispatch_get_main_queue();
            dispatch_async(mainQueue, ^{

                [self presentViewController:interstitialAdViewController animated:YES completion:^{}];

            });

            [interstitialAdViewController release];
        }
    }];
}

- (void)viewWillDisappear:(BOOL)animated
{
    [[AdService sharedAdService] clearAdForSlotName:@"Main Interstitial"];

    [super viewWillDisappear:animated];
}
4

2 回答 2

5

您链接的答案是关于 viewDidDisappear 的行为,而不是 viewDidAppear。您链接的文档中的注释说,当带有弹出框的子视图控制器被解除时,不会在当前视图控制器上调用 viewDidAppear 。

在您的情况下,您正在呈现没有弹出窗口的子视图控制器,所以我阅读了文档的注释(以“证明规则的例外”的方式)说当子视图控制器是时,应该在呈现视图控制器上调用 viewDidAppear在你的情况下被解雇。

我认为当子视图控制器出现时,您应该在父视图控制器上看到 viewDidDisappear 被调用。我在一个最小的测试应用程序中尝试了这个,使用这个代码来展示一个子视图控制器:

Parent *parent = [[Parent alloc] initWithNibName:nil bundle:nil];
[self.window setRootViewController:parent];
double delayInSeconds = 2.0;
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
    Child *child = [[Child alloc] initWithNibName:nil bundle:nil];
    [parent presentViewController:child animated:YES completion:^{
        NSLog(@"Child presented");
        double delayInSeconds = 2.0;
        dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
        dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
            [parent dismissViewControllerAnimated:YES completion:^{
                NSLog(@"Child dismissed");
            }];
        });
    }];
});

并且,在 Parent 和 Child 类中记录对各种外观相关回调的调用。这是我看到的顺序:

使父视图控制器成为根视图控制器:

Parent viewWillAppear:
Parent viewDidAppear:

呈现子视图控制器:

Parent viewWillDisappear:
Child viewWillAppear:
Child viewDidAppear:
Parent viewDidDisappear:
Child presented

关闭子视图控制器:

Child viewWillDisappear:
Parent viewWillAppear:
Parent viewDidAppear:
Child viewDidDisappear:
Child dismissed

所以这对我来说似乎在内部是一致的——当子视图控制器出现时父级得到消失调用,当子视图控制器被关闭时调用外观。这仍然与 viewWillAppear 的记录行为一致,因为我没有向子视图控制器显示弹出框。

恐怕我不知道为什么当您的子视图控制器出现时您没有收到 viewWill/DidDisappear 调用。

于 2013-05-26T18:39:42.640 回答
3

我经历过同样的行为。该viewDidAppear:方法在关闭呈现的视图控制器时被调用。但是,有一条出路。在控制器中创建一个布尔变量,将其设置为NOinviewDidLoad并将其签入viewDidAppear:. 然后将其设置为YES. 像这样:

#import "MyViewController.h"

@implementation MyViewController {
    BOOL alreadyShown;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    alreadyShown = NO;
}

- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    if (!alreadyShown) {
        alreadyShown = YES;
        //do your stuff
    }
}

@end
于 2013-05-26T18:42:08.560 回答