如何从故事板加载初始窗口控制器?

时间:2015-06-22 07:42:33

标签: macos swift cocoa storyboard

我经历了许多任务,但他们都没有抨击我的查询。 我试图以编程方式加载初始窗口 这就是我所做的。

我已将main.swift添加为 -

import Cocoa

private func runApplication(
    application: NSApplication         = NSApplication.sharedApplication(),
    delegate: NSApplicationDelegate?   = AppDelegate(),
    bundle: NSBundle                   = NSBundle.mainBundle(),
    nibName: String                    = "MainMenu",
    var topLevelObjects: NSArray?      = nil) {
        setApplicationDelegate(application, delegate)
}

private func setApplicationDelegate(application: NSApplication, delegate: NSApplicationDelegate?) -> NSApplication {
    if let delegate = delegate {
        application.delegate = delegate
    }
    return application
}

runApplication()

Appdelegate.swift是 -

import Cocoa

//@NSApplicationMain
class AppDelegate: NSObject, NSApplicationDelegate {
    var appControl:AppFlow?
    func applicationDidFinishLaunching(aNotification: NSNotification) {

    }

    func applicationWillTerminate(aNotification: NSNotification) {
        // Insert code here to tear down your application
    }
    override init() {
        //
        self.appControl = AppFlow()
        super.init()
    }



}

在AppFlow中,我试图从故事板加载窗口控制器.-

import Cocoa

class AppFlow{
    let initialStoryBoard:NSStoryboard?
    override init() {
        self.initialStoryBoard = NSStoryboard(name: "Main" , bundle : nil)
        super.init()
        var windowController = (self.initialStoryBoard?.instantiateControllerWithIdentifier("mainWindow")) as! NSWindowController
        windowController.window?.makeKeyAndOrderFront(nil)     
    }

}

但是我无法启动初始窗口控制器和视图控制器。应用程序自动启动和终止,不向用户显示任何窗口。

我做错了什么?谢谢你的帮助。

2 个答案:

答案 0 :(得分:2)

以下是为了在没有属性MainMenu和函数@NSApplicationMain

的情况下以编程方式加载故事板(以及NSApplicationMain(_, _))的初始窗口所做的操作

文件: AppConfig.swift (Swift 4)

struct AppConfig {

   static var applicationClass: NSApplication.Type {
      guard let principalClassName = Bundle.main.infoDictionary?["NSPrincipalClass"] as? String else {
         fatalError("Seems like `NSPrincipalClass` is missed in `Info.plist` file.")
      }
      guard let principalClass = NSClassFromString(principalClassName) as? NSApplication.Type else {
         fatalError("Unable to create `NSApplication` class for `\(principalClassName)`")
      }
      return principalClass
   }

   static var mainStoryboard: NSStoryboard {
      guard let mainStoryboardName = Bundle.main.infoDictionary?["NSMainStoryboardFile"] as? String else {
         fatalError("Seems like `NSMainStoryboardFile` is missed in `Info.plist` file.")
      }

      let storyboard = NSStoryboard(name: NSStoryboard.Name(mainStoryboardName), bundle: Bundle.main)
      return storyboard
   }

   static var mainMenu: NSNib {
      guard let nib = NSNib(nibNamed: NSNib.Name("MainMenu"), bundle: Bundle.main) else {
         fatalError("Resource `MainMenu.xib` is not found in the bundle `\(Bundle.main.bundlePath)`")
      }
      return nib
   }

   static var mainWindowController: NSWindowController {
      guard let wc = mainStoryboard.instantiateInitialController() as? NSWindowController else {
         fatalError("Initial controller is not `NSWindowController` in storyboard `\(mainStoryboard)`")
      }
      return wc
   }
}

档案 main.swift (Swift 4)

// Making NSApplication instance from `NSPrincipalClass` defined in `Info.plist`
let app = AppConfig.applicationClass.shared

// Configuring application as a regular (appearing in Dock and possibly having UI)
app.setActivationPolicy(.regular)

// Loading application menu from `MainMenu.xib` file.
// This will also assign property `NSApplication.mainMenu`.
AppConfig.mainMenu.instantiate(withOwner: app, topLevelObjects: nil)

// Loading initial window controller from `NSMainStoryboardFile` defined in `Info.plist`.
// Initial window accessible via property NSWindowController.window
let windowController = AppConfig.mainWindowController
windowController.window?.makeKeyAndOrderFront(nil)

app.activate(ignoringOtherApps: true)
app.run()

关于MainMenu.xib档案的说明:

Xcode应用程序模板使用Application Scene创建包含Main Menu的故事板。目前似乎无法以编程方式从Main Menu加载Application Scene。但是有Xcode文件模板Main Menu,它创建了MainMenu.xib文件,我们可以通过编程方式加载。

答案 1 :(得分:0)

这不是您启动(和维护)应用程序主运行循环的方式。见@NSApplicationMain。这导致主运行循环被设置并运行直到它终止。我们不再需要main.swift文件,因为您可以直接将其放入您的应用代理人的文件中。

import Cocoa

@NSApplicationMain
class AppDelegate: NSObject, NSApplicationDelegate {

    @IBOutlet weak var window: NSWindow!


    func applicationDidFinishLaunching(aNotification: NSNotification) {
        // Insert code here to initialize your application
    }

}

Xcode的新应用程序项目模板为您完成此任务。