我整理了一个快速的 Swift 小操场,展示了我是如何解决这个问题的。它涉及一个帮助函数,它为窗口创建变量,进行一些设置,然后将内容设置为 SwiftUI 视图并将变量传递给该 SwiftUI 视图。这为视图提供了对包含它的窗口的引用,让它在该窗口上调用 close()。
import SwiftUI
func showWindow() {
var windowRef:NSWindow
windowRef = NSWindow(
contentRect: NSRect(x: 0, y: 0, width: 100, height: 100),
styleMask: [.titled, .closable, .miniaturizable, .fullSizeContentView],
backing: .buffered, defer: false)
windowRef.contentView = NSHostingView(rootView: MyView(myWindow: windowRef))
windowRef.makeKeyAndOrderFront(nil)
}
struct MyView: View {
let myWindow:NSWindow?
var body: some View {
VStack{
Text("This is in a separate window.")
HStack{
Button(action:{
showWindow()
}) {
Text("Open another window")
}
Button(action:{
self.myWindow!.close()
}) {
Text("Close this window")
}
}
}
.padding()
}
}
showWindow()
我将 myWindow 变量设为可选项,这样您就可以在 Xcode 的预览中传递 nil,如下所示:
struct MyView_Previews: PreviewProvider {
static var previews: some View {
MyView(myWindow: nil)
}
}
编辑添加:我意识到我没有直接回答 OP 提出的问题:我应该在按钮操作中写什么?如果登录成功,我如何以及在哪里调用登录函数以及该函数将如何更改窗口?
我有一个具有类似模式的应用程序(登录,然后显示来自服务器的数据的不同窗口),这就是为什么我必须找出上面操场的代码。我建立了一个类来表示与我正在使用的服务的连接。如果遇到错误,该对象的初始化程序可能会抛出,从而导致抛出错误并且没有对象。
对于按钮操作,我使用这样的代码(我实际上并没有运行这个确切的代码,所以可能会有错误):
Button(action: {
let loginResult = Result {try connectToServer(self.address, username: self.username, password: self.password)}
switch loginResult {
case .failure(let error):
print(error.localizedDescription)
let loginAlert = NSAlert(error: error)
loginAlert.beginSheetModal(for: self.myWindow, completionHandler: {...})
case .success(let serverConnection):
showContentWindow(serverConnection)
self.myWindow.close()
}
}) {
Text("Login")
}
showContentWindow 是一个辅助函数,类似于上面操场中的那个。它接受表示 API 连接的对象,然后将其传递给它用作窗口内容的 SwiftUI 视图,就像上面将窗口传递给窗口内的视图一样。
您显然可以通过多种方式处理错误。上面的代码与我使用的代码很接近(尽管我现在没有它在我面前),并为我提供了网络超时等错误的免费本地化描述。不过,错误处理细节远远超出了本文的范围。
重要的是在成功时,它调用函数打开新窗口,然后在自己的窗口上调用close。漂亮又简单。