gpt4 book ai didi

ios - UINavigationController 状态恢复(无 Storyboard)

转载 作者:IT王子 更新时间:2023-10-29 08:11:40 25 4
gpt4 key购买 nike

我一直在研究状态恢复。在下面的代码中,UITableViewController 的滚动位置得到恢复,但是,如果我要进入详细 View (将 MyViewController 的实例推到导航堆栈上),当应用程序重新启动时,它总是返回到第一个 View 导航堆栈中的 Controller (即 MyTableViewController)。有人能帮我恢复到正确的 View Controller (即 MyOtherViewController)吗?

AppDelegate.m

- (BOOL)launchWithOptions:(NSDictionary *)launchOptions
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
// Override point for customization after application launch.


MyTableViewController *table = [[MyTableViewController alloc] initWithStyle:UITableViewStylePlain];
table.depth = 0;
UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:table];
navCon.restorationIdentifier = @"navigationController";

self.window.rootViewController = navCon;

self.window.backgroundColor = [UIColor whiteColor];
[self.window makeKeyAndVisible];

});

return YES;
}

- (BOOL)application:(UIApplication *)application willFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
return [self launchWithOptions:launchOptions];
}

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
return [self launchWithOptions:launchOptions];
}

MyTableViewController.m

- (id)initWithStyle:(UITableViewStyle)style
{
self = [super initWithStyle:style];
if(self)
{
self.restorationIdentifier = @"master";
}
return self;
}

- (void)viewDidLoad
{
[super viewDidLoad];
self.title = @"Master";
self.tableView.restorationIdentifier = @"masterView";
}

#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
return 5;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
return 10;
}

- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section
{
return [NSString stringWithFormat:@"Section %d", section];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *CellIdentifier = @"Cell";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if(!cell)
{
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
}

cell.textLabel.text = [NSString stringWithFormat:@"%d", indexPath.row];

return cell;
}

#pragma mark - Table view delegate

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
MyOtherViewController *vc = [[MyOtherViewController alloc] initWithNibName:nil bundle:nil];
[self.navigationController pushViewController:vc animated:YES];
}

MyOtherViewController.m

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) {
self.restorationIdentifier = @"detail";
}
return self;
}

- (void)viewDidLoad
{
[super viewDidLoad];
self.title = @"Detail";
self.view.backgroundColor = [UIColor redColor];
self.view.restorationIdentifier = @"detailView";
}

最佳答案

因为您是在代码中创建细节 View Controller ,而不是从 Storyboard 实例化它,所以您需要实现一个恢复类,以便系统恢复过程知道如何创建细节 View Controller 。

恢复类实际上只是一个知道如何从恢复路径创建特定 View Controller 的工厂。你实际上不必为此创建一个单独的类,我们可以在 MyOtherViewController 中处理它:

在 MyOtherViewController.h 中,实现协议(protocol):UIViewControllerRestoration

然后当你在MyOtherViewController.m中设置restorationID时,同时设置恢复类:

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) {
self.restorationIdentifier = @"detail";
self.restorationClass = [self class]; //SET THE RESTORATION CLASS
}
return self;
}

然后你需要在恢复类(MyOtherViewController.m)中实现这个方法

    +(UIViewController *) viewControllerWithRestorationIdentifierPath:(NSArray *)identifierComponents coder:(NSCoder *)coder
{
//At a minimum, just create an instance of the correct class.
return [[MyOtherViewController alloc] initWithNibName:nil bundle:nil];
}

这使您能够创建详细 View Controller 并将其推送到导航堆栈的恢复子系统。 (你最初问的是什么。)

扩展示例以处理状态:

在真实的应用程序中,您需要保存状态并处理恢复它...我将以下属性定义添加到 MyOtherViewController 来模拟这一点:

@property (nonatomic, strong) NSString *selectedRecordId;

我还修改了 MyOtherViewContoller 的 viewDidLoad 方法,将详细信息标题设置为用户选择的任何记录 ID:

- (void)viewDidLoad
{
[super viewDidLoad];
// self.title = @"Detail";
self.title = self.selectedRecordId;
self.view.backgroundColor = [UIColor redColor];
self.view.restorationIdentifier = @"detailView";
}

为了使示例正常工作,我在最初从 MyTableViewController 推送详细信息 View 时设置了 selectedRecordId:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
MyOtherViewController *vc = [[MyOtherViewController alloc] initWithNibName:nil bundle:nil];
vc.selectedRecordId = [NSString stringWithFormat:@"Section:%d, Row:%d", indexPath.section, indexPath.row];
[self.navigationController pushViewController:vc animated:YES];
}

另一个缺失的部分是 MyOtherViewController 中的方法,您的详细 View ,用于保存详细信息 Controller 的状态。

-(void)encodeRestorableStateWithCoder:(NSCoder *)coder
{
[coder encodeObject:self.selectedRecordId forKey:@"selectedRecordId"];
[super encodeRestorableStateWithCoder:coder];
}

-(void)decodeRestorableStateWithCoder:(NSCoder *)coder
{
self.selectedRecordId = [coder decodeObjectForKey:@"selectedRecordId"];
[super decodeRestorableStateWithCoder:coder];
}

现在这实际上还不太奏效。这是因为在 decoreRestorablStateWithCoder 方法之前,“ViewDidLoad”方法在恢复时被调用。因此,在显示 View 之前不会设置标题。为了解决这个问题,我们要么在 viewWillAppear: 中设置 Detail View Controller 的标题,要么我们可以修改 viewControllerWithRestorationIdentifierPath 方法以在它创建类时恢复状态,如下所示:

+(UIViewController *) viewControllerWithRestorationIdentifierPath:(NSArray *)identifierComponents coder:(NSCoder *)coder
{
MyOtherViewController *controller = [[self alloc] initWithNibName:nil bundle:nil];
controller.selectedRecordId = [coder decodeObjectForKey:@"selectedRecordId"];
return controller;

}

就是这样。

一些其他注意事项...

即使我没有看到代码,我假设您在 App Delegate 中执行了以下操作?

-(BOOL) application:(UIApplication *)application shouldSaveApplicationState:(NSCoder *)coder
{
return YES;
}
- (BOOL) application:(UIApplication *)application shouldRestoreApplicationState:(NSCoder *)coder
{
return YES;
}

在模拟器中运行演示代码时,我看到了一些不稳定的地方,它正确地保留了滚动偏移。它似乎偶尔对我有用,但并非一直有效。我怀疑这可能是因为 MyTableViewController 的真正恢复也应该使用 restorationClass 方法,因为它是在代码中实例化的。不过我还没有尝试过。

我的测试方法是通过返回 springboard 将应用程序置于后台(需要保存应用程序状态。),然后从 XCode 中重新启动应用程序以模拟干净的启动。

关于ios - UINavigationController 状态恢复(无 Storyboard),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14376786/

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