如何从其他类(NSViewController 的子类)获取主窗口(App Delegate)?

Posted

技术标签:

【中文标题】如何从其他类(NSViewController 的子类)获取主窗口(App Delegate)?【英文标题】:How to get Main Window (App Delegate) from other class (subclass of NSViewController)? 【发布时间】:2011-10-01 12:39:09 【问题描述】:

我正在尝试从其他类(即 NSViewController 的子类)更改我的 Windows 内容。我正在尝试下面的代码,但它什么也没做。

[NSApplication sharedApplication]mainWindow]setContentView:[self view]]; //code in NSViewController

[NSApplication sharedApplication]mainWindow] // returns null

我尝试添加

[window makeMainWindow];

在 App Delegate 类中,但无济于事。

我错过了什么吗?

附:另外我正在使用下面的代码来调用我班级中的任何委托函数,

 [(appDelegate *) [[NSApplication sharedApplication]delegate]MyMethod];

但我想知道是否有更好的方法,但不导入委托类。像这样的

[[NSApplication sharedApplication]delegate]MyMethod];

(它给出警告)

【问题讨论】:

【参考方案1】:

对于mainWindow method the docs say:

如果应用程序的 nib 文件尚未完成加载、接收器未处于活动状态或应用程序处于隐藏状态,此方法可能会返回 nil。

我刚刚创建了一个快速测试应用程序,并放置了以下代码:

NSLog(@"%@", [[NSApplication sharedApplication] mainWindow]);

进入我的applicationDidFinishLaunching:aNotification 方法,并进入我连接到应用程序主窗口中的按钮的操作方法。

在启动时,mainWindow 为 nil,但当我单击按钮时(在一切启动并运行并显示后),mainWindow 不再为 nil。

NSApplication 提供了其他可能对您有用的方法:

- windows - 所有窗口的数组; – keyWindow - 给出正在接收键盘输入(或 nil)的窗口; – windowWithWindowNumber: - 返回与窗口编号对应的窗口 - 如果您知道要替换其内容的窗口编号,则可以使用它; – makeWindowsPerform:inOrder: - 向每个窗口发送一条消息 - 您可以使用它来测试每个窗口,看看它是否是您感兴趣的窗口。

关于在delegate 上调用方法,你所说的给出警告对我来说很好。例如,这可以在没有警告的情况下工作:

NSLog(@"%@", [[[NSApplication sharedApplication]delegate] description]);

您收到的警告究竟是什么?您是否尝试调用不存在的方法?

【讨论】:

非常感谢。我只有一个窗口。所以可能我会使用 windows 来进入 Array,并会使用 Array 中的第一个对象。关于委托的方法,您是否从不同的类(如视图控制器)尝试过?当我调用它时,它可以正常工作,但它会发出警告,该委托可能不会响应该方法。 我明白你现在关于代表电话的意思了。不幸的是,[[NSApplication sharedApplication] delegate] 消息总是会返回一个 NSApplicationDelegate 类型的对象。如果您想向它发送特定于您的应用程序委托的消息,则必须将其转换为您的应用程序委托。您可能想将此作为一个单独的问题提出? 我认为唯一的方法是将 NSApplicationDelegate 转换为特定的委托,并将其导入类中。谢谢。 还有另一种方式!你可以 [[NSApplication sharedApplication]delegate]performSelector:@selector(selectorName)];如果您知道委托中的选择器名称,则无需包含 appdelegate 标头和类型转换。这会奏效。 在 Swift Playground 中怎么样?如何获得非零主窗口?【参考方案2】:

与 MacOS 的斗争才发现了这一点。

苹果的报价:

主窗口

房产

应用程序的主窗口。 (只读)

讨论

当应用程序的故事板或 nib 文件尚未完成加载时,此属性中的值为 nil。当应用不活动隐藏时,它也可能为 nil。

如果您的应用程序中只有一个窗口(这是最常用的情况),请使用下一个代码:

NSWindow *mainWindow = [[[NSApplication sharedApplication] windows] objectAtIndex:0];

保证它不会是零,如果应用程序有窗口。

【讨论】:

我正在评论 3 年前的一个问题,但这似乎也不起作用。它不应该返回一些东西吗,因为代码进入的 ViewController 在它尝试获取窗口时已经完成加载? 您将此代码粘贴到 viewDidLoad 中它应该可以工作。当然,如果您的应用程序中没有窗口,那么它将为零。 感谢您的回复。它在 tableView 方法之一中,因为它是 NSTableView 的委托。所以我不知道为什么它不起作用,窗口显示内容。 我很确定你不能保证它不会是 nil。 当然,如果应用程序没有任何窗口,它将是 nil...相应地编辑答案。【参考方案3】:

如果尚未设置 window 属性,请尝试延迟操作,直到应用完成加载,如下所示:

[myObject performSelector:@selector(theSelector) withObject:nil afterDelay:0.1];

【讨论】:

【参考方案4】:

获取主窗口(如果存在)的 Swift 风格的方法

应用程序主窗口

guard let window = NSApplication.shared.mainWindow,
else 
    // handle no main window present

// ... access window here

应用程序窗口数组

let windowArray: [NSWindow] = NSApplication.shared.windows
guard windowArray.count > 0 else 
    // hand case where no windows are present

let window = windowArray[0]
// ... access window here

【讨论】:

以上是关于如何从其他类(NSViewController 的子类)获取主窗口(App Delegate)?的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Mac OS X 应用程序中从一个 NSViewcontroller 推送到另一个 NSViewcontroller?

如何从app delegate访问NSViewController?

如何从 OS X 上的框架实例化 NSViewController(即获取正确的包)

从 NSTableView 获取数据到 NSViewController

NSViewController 没有被创建

从 NSViewController 在 OS X 上加载 nib