18

我一直在玩弄状态恢复。在下面的代码中,UITableViewController 的滚动位置被恢复,但是,如果我要点击进入详细视图(将 MyViewController 的实例推送到导航堆栈上),当应用程序重新启动时,它总是返回到第一个视图导航堆栈中的控制器(即 MyTableViewController)。有人能帮我恢复到正确的视图控制器(即 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";
}
4

3 回答 3

18

因为您是在代码中创建详细视图控制器,而不是从 Storyboard 中实例化它,所以您需要实现一个恢复类,因此系统恢复过程知道如何创建详细视图控制器。

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

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

然后当你在 MyOtherViewController.m 中设置 restoreID 时,还要设置恢复类:

- (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];
}

这使您能够创建恢复子系统并将详细视图控制器推送到导航堆栈上。(你最初问的是什么。)

扩展示例以处理状态:

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

@property (nonatomic, strong) NSString *selectedRecordId;

我还修改了 MyOtherViewContoller 的 viewDidLoad 方法,将 Detail 标题设置为用户选择的任何记录 Id:

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

为了使示例工作,我在最初从 MyTableViewController 推送详细视图时设置了 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 中的方法,您的详细视图,用于保存 Detail 控制器的状态。

-(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];
}

现在这实际上还没有完全奏效。这是因为“ViewDidLoad”方法在 deoreRestorablStateWithCoder 方法之前在恢复时被调用。因此,在显示视图之前不会设置标题。为了解决这个问题,我们可以在 viewWillAppear: 中设置 Detail 视图控制器的标题,或者我们可以修改 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 也应该使用 restoreClass 方法,因为它是在代码中实例化的。但是我还没有尝试过。

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

于 2013-01-23T01:35:15.770 回答
1

您的代码的唯一问题是导航控制器这两行

UINavigationController *navCon = [[UINavigationController alloc] initWithRootViewController:table];
    navCon.restorationIdentifier = @"navigationController";

我不知道为什么导航控制器永远不会得到它的恢复类。我有同样的问题。我找到了一些替代解决方案来在故事板中创建一个独立的导航控制器并使用它。

这是代码

UIStoryboard* storyboard = [UIStoryboard storyboardWithName:@"MainStoryboard" 
    bundle:nil];
UINavigationController* navigationController = [storyboard
    instantiateViewControllerWithIdentifier:
    @"UINavigationController"];

navigationController.viewControllers = @[myController];

它会正常工作。

只需遵循此http://petr-pavlik.squarespace.com/blog/2013/6/16/stare-restoration-without-storyboard

于 2013-06-28T11:20:47.147 回答
0

由于您是在代码中而不是通过 Storyboard 执行此操作,因此您不仅需要提供一个细节视图控制器,restorationIdentifier还需要提供一个restorationClass细节视图控制器。

您可以将其保留为未分配,在这种情况下-(UIViewController *)application:(UIApplication *)application viewControllerWithRestorationIdentifierPath:(NSArray *)identifierComponents coder:(NSCoder *)coder,会在应用程序委托上调用以创建对象(带有restorationIdentifier但没有的视图控制器restorationClass)。

尝试以下操作(请注意UIViewControllerRestoration协议):

@interface MyViewController ()<UIViewControllerRestoration>
@end

@implementation

- (id)init
{
    self = [super init];
    if (self) {
        // Custom initialization
    }

    if( [self respondsToSelector:@selector(restorationIdentifier)] ){
        self.restorationIdentifier = @"DetailViewController";
        self.restorationClass = [self class];
    }
    return self;
}

#pragma mark - State restoration

+ (UIViewController *)viewControllerWithRestorationIdentifierPath:(NSArray *)identifierComponents coder:(NSCoder *)coder
{
    UIViewController *viewController = [[self alloc] init];

    return viewController;
}

@end

另请注意,这是一个非常简单的示例,通常您可能会有更多有趣的事情发生-viewControllerWithRestorationIdentifierPath:coder:

于 2013-05-05T21:44:17.250 回答