gpt4 book ai didi

ios - 应用程序委托(delegate)检查所有 View Controller 转换何时完成

转载 作者:行者123 更新时间:2023-11-29 10:40:43 25 4
gpt4 key购买 nike

我希望我的应用能够响应发送到我设备的远程推送。我想做的是呈现一个 View Controller 作为对推送通知的响应。

当应用程序运行时,这样做没有问题。从应用程序委托(delegate)开始,我首先获取 Root View Controller 并一直向上,以便找到可见 View Controller 并在其上显示一个新 View Controller 。像魅力一样工作。

然而,当应用程序启动时,我首先显示 Root View Controller ,然后检查是否有任何缓存数据存储在设备上,如果有,它会加载一个新的 View Controller 来呈现它。有时需要 2 秒,有时需要 7 秒。当应用程序完成呈现新 View Controller 时,没有办法告诉操作将花费多长时间。因此,如果我想在启动后立即呈现 View Controller ,情况就会变得棘手。

目前我使用的是依赖于 dispatch_after 片段的 hacky 版本,它在应用启动后大约 5 秒触发演示。它适用于 95% 的情况,但我不喜欢这个解决方案。

所以我的问题是:当所有模式转换都完成并且我的应用程序处于“稳定的最终状态”(这样我可以获得最顶层的 View Controller 和呈现在它上面没有伤害)状态?

如果有任何帮助,我可以粘贴一些我已经在使用的代码。

编辑
我被要求粘贴一些代码,说明我如何从应用程序委托(delegate)中呈现 View Controller 。当调用 application:didFinishLaunchingWithOptions: 并且 launchOptions 字典中有一个 UIApplicationLaunchOptionsRemoteNotificationKey 条目时,将调用下面的代码段。

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5.0f * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
UIViewController *controller = [self topViewControllerFromController:self.window.rootViewController];
NSInteger messageID = [userInfo[@"m_id"] integerValue];

MessageViewController *mvc = [[MessageViewController alloc] initWithMessageID:messageID];
mvc.modalTransitionStyle = UIModalTransitionStyleCoverVertical;

[controller presentViewController:mvc animated:YES completion:nil];
});

topViewController 非常简单:

- (UIViewController *)topViewControllerFromController:(UIViewController *)controller
{
if (controller.presentedViewController)
{
UIViewController *presentedViewController = controller.presentedViewController;
return [self topViewControllerFromController:presentedViewController];
}
else if ([controller isKindOfClass:[UINavigationController class]])
{
UINavigationController *navigationController = (UINavigationController *)controller;
return [self topViewControllerFromController:navigationController.visibleViewController];
}
else if ([controller isKindOfClass:[UITabBarController class]])
{
UITabBarController *tabBarController = (UITabBarController *)controller;
return [self topViewControllerFromController:tabBarController.selectedViewController];
}
else
{
return controller;
}
}

现在演示文稿和其他东西可以正常工作,没有任何问题。我只需要知道所有模态转换何时完成,即。当该应用适合展示另一个应用时。

最佳答案

如果您有一个固定的 View 层次结构来表示要显示的最终 View ,您可以:

A) 最简单,从您拥有的任何“加载链末端” View Controller 的 [edit:] viewDidAppear: 方法发布通知。顺便说一句,您应该在一个 block 中使用对 self 的弱引用,就像您显示的那样(见下文,在 block 中使用 selfRef,而不是关键字 self)。如果你不这样做,它会生成一个保留周期,你的应用程序将保留内存(技术上不是泄漏,但有类似的效果)。在这种情况下可能不会太重要,因为您的应用程序不太可能关闭 topViewController,因此在这种情况下它不太可能对内存产生净负面影响,但作为风格点它仍然很重要。 :

在你的 appDelegate 中的某处(例如在 application:didFinishLaunching:withOptions: 方法中)

NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
__weak UIViewController *selfRef = self;

[center addObserverForName:@"AllViewsLoaded" object:nil queue:nil usingBlock:^(NSNotification *note)
{
UIViewController *controller = [selfRef topViewControllerFromController: selfRef.window.rootViewController];
NSInteger messageID = [userInfo[@"m_id"] integerValue];

MessageViewController *mvc = [[MessageViewController alloc] initWithMessageID:messageID];
mvc.modalTransitionStyle = UIModalTransitionStyleCoverVertical;

[controller presentViewController:mvc animated:YES completion:nil];
}];

然后在 viewDidAppear: 任何“行尾” View Controller 的方法中:

[[NSNotificationCenter defaultCenter] postNotification:[NSNotification notificationWithName:@"AllViewsLoaded" object:nil]];

B) 或者,您可以定义一个协议(protocol) - 例如在最终 View Controller 的 .h 文件中:

@protocol FinalViewLoadedAndDisplayedProtocol
- (void)viewFinallyLoaded:(UIViewController*)finalViewControllerThatLoaded;
@end

在您的应用委托(delegate)中定义方法:

- (void)viewFinallyLoaded:(UIViewController*)finalViewControllerThatLoaded {
UIViewController *controller = [self topViewControllerFromController: self.window.rootViewController];
NSInteger messageID = [userInfo[@"m_id"] integerValue];

MessageViewController *mvc = [[MessageViewController alloc] initWithMessageID:messageID];
mvc.modalTransitionStyle = UIModalTransitionStyleCoverVertical;

[controller presentViewController:mvc animated:YES completion:nil];
}

将协议(protocol)支持添加到您的 App delegate .h 文件中。然后在最终 View Controller 的 .m 文件中:

- (void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
MainClass *appDelegate = (id)[[UIApplication sharedApplication] delegate];

if ([appDelegate respondsToSelector:@selector(viewFinallyLoaded:)]) {
[self.viewDidLoadDelegate viewFinallyLoaded:self];
}
}

关于ios - 应用程序委托(delegate)检查所有 View Controller 转换何时完成,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24692028/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com