0

我正在编写一个 Cocoa 应用程序,并希望该应用程序作为一种向导工作。因此,在主窗口中,我有一个自定义视图,它与用户交互,并在用户逐步完成向导的各个阶段时从登录屏幕更改为设备激活屏幕。我目前已经覆盖了 WizardViewController 的 awakeFromNib 方法:

- (void)awakeFromNib{
   //If no redirect request save, add first view: ID Login
   NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
   NSString *tokenRequest = [defaults objectForKey:@"redirectRequestToken"];
   if (!tokenRequest){
       SignInWithIDViewController *signInViewController = [[SignInWithIDViewController alloc] initWithNibName:@"SignInWithIDViewController" bundle:nil];
       [wizardView addSubview:[signInViewController view]];
   } else {
    NSLog(@"Have already logged in.");
   }
}

照原样,SignInIDViewController 中的 initWithNibName 被调用了两次,一次是由我显式调用,另一次是在加载视图时(可能是通过 loadView)。但是,如果我只是调用 init,那么 initWithNib 名称只会调用一次,但会加载错误的 xib 文件(属于 DeviceActivationViewController 类)。我似乎无法弄清楚我做错了什么,因为 signInViewController 不应该被初始化两次,但我需要在 IB 中显示正确的 xib 文件。

我目前在此类中唯一不是用户界面 IBAction 的方法是生成的 initWithNibName 方法加上添加的 NSLog 语句。

4

1 回答 1

1

我认为在 IB(蓝色立方体)中创建对象并在代码中实例化它们是问题所在。如果您在 IB 中为它们创建了对象,那么它们将在 awakeFromNib 中实例化,您也不应该在代码中对它们调用 alloc init ——这将创建一个新实例。

我在 OSX 中使用视图控制器方面没有很多经验,但您似乎无法将 IBActions 连接到视图控制器(作为文件的所有者)。我使它工作的方式是子类化自定义视图(当您添加视图控制器时为您创建),将该视图的类更改为您的新子类,并将操作方法​​放入该类中。看起来这应该由视图控制器处理,但我认为它不起作用与视图控制器不在 OSX 的响应者链中有关(而我认为它在 iOS 中)。

编辑后:在绕道内存管理问题之后,我想我找到了最好的方法。您可以并且可能应该(以符合 Apple 的 MVC 范例)将按钮方法放在视图控制器类中,而不是像我上面所说的那样放在视图中。您实际上可以将 IBActions 连接到视图控制器(作为文件的所有者),您只需要确保在代码中实例化它时保留视图控制器。为此,您需要在要在其中实例化 SignInViewController 类的任何类中将 signInViewController 设为属性,并在属性声明中使用“retain”。那么您不需要(也不应该)在 IB 中创建任何蓝色立方体。

于 2012-06-01T16:12:35.343 回答