簡體   English   中英

Swift 在 macOS 中以編程方式打開新窗口

[英]Swift open new window programmatically in macOS

在使用 macOS 的 Swift 中:通過在AppDelegate 中刪除@NSApplicationMain (並創建NSWindowController的子類),我以編程方式創建主窗口,而不使用故事板等:

//@NSApplicationMain
class AppDelegate: NSObject, NSApplicationDelegate {

    var window: NSWindow!
    var viewController: NSViewController!
    var windowController: NSWindowController!


    func configMainWindow(_ viewController: NSViewController) {
        window = NSWindow(contentRect: NSRect(x: 0, y: 0, width: 800, height: 600),
                      styleMask: [NSWindow.StyleMask.closable, NSWindow.StyleMask.titled, NSWindow.StyleMask.resizable, NSWindow.StyleMask.miniaturizable],
                      backing: NSWindow.BackingStoreType.buffered,
                      defer: false)
        window.title = "My App"
        window.setFrameAutosaveName("My App")
        window.center()
        window.isOpaque = false
        window.isMovableByWindowBackground = true
        window.backgroundColor = NSColor.white
        window.makeKeyAndOrderFront(nil)
        window.contentViewController = viewController
        windowController = WindowController(window: window)
    }


    func applicationDidFinishLaunching(_ aNotification: Notification) {
        viewController = ViewController()
        configMainWindow(viewController)
    }
}

windowController附加了一個toolbarstatusBarmenuBar :(只有 menuBar 從NIB加載。類MainMenuAction處理菜單選擇。)

class WindowController: NSWindowController, NSWindowDelegate {

    var toolbarController = ToolbarController()
    var statusBarController = StatusBarController()
    var mainMenuAction: MainMenuAction?


    override init(window: NSWindow?) {
        super.init(window: window)
        window?.toolbar = toolbarController.toolbar
        window?.delegate = self

        var topLevelObjects: NSArray? = []
        Bundle.main.loadNibNamed("MainMenu", owner: self, topLevelObjects: &topLevelObjects)
        NSApplication.shared.mainMenu = topLevelObjects?.filter { $0 is NSMenu }.first as? NSMenu
        self.mainMenuAction = MainMenuAction.shared
    }

    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }


    override func windowDidLoad() {
        if let window = window {
            if let view = window.contentView {
                view.wantsLayer = true
                window.titleVisibility = .hidden
                window.titlebarAppearsTransparent = true
                window.backgroundColor = .white
            }
        }
    }
}

另外,我需要添加一個main.swift文件:(感謝提醒我,apodidae)

let delegate = AppDelegate()
NSApplication.shared.delegate = delegate
_ = NSApplicationMain(CommandLine.argc, CommandLine.unsafeArgv)

我試過:

    let vc = NSViewController()
    let win = configWindow(vc, windowWidth: 420, windowHeight: 673)
    let wc = NSWindowController(window: win)
    wc.window?.present
    vc.view.window?.contentViewController = vc

我從 AppDelegate 復制了 configMainWindow 方法,以創建允許我指定大小和 vc 的 configWindow。


但是如何打開一個具有自定義大小樣式的新窗口(從新類中的某些方法 - 以編程方式)?

請提供代碼示例。

以下演示創建了由自定義類中的方法調用的第二個窗口。 它可以通過添加“swift.main”文件並用以下代碼替換 AppDelegate 在 Xcode 中運行:

import Cocoa

class Abc : NSObject {
var panel: NSPanel!

func buildWnd2() {
 let _panelW : CGFloat = 200
 let _panelH : CGFloat = 200

 panel = NSPanel(contentRect:NSMakeRect(9300, 1300, _panelW, _panelH), styleMask:[.titled, .closable, .utilityWindow],
  backing:.buffered, defer: false)         
 panel.isFloatingPanel = true
 panel.title = "NSPanel"
 panel.orderFront(nil)
}
}

let abc = Abc()

class AppDelegate: NSObject, NSApplicationDelegate {
var window:NSWindow!

@objc func myBtnAction(_ sender:AnyObject ) {
 abc.buildWnd2()
}

func buildMenu() {
 let mainMenu = NSMenu()
 NSApp.mainMenu = mainMenu
 // **** App menu **** //
 let appMenuItem = NSMenuItem()
 mainMenu.addItem(appMenuItem)
 let appMenu = NSMenu()
 appMenuItem.submenu = appMenu
 appMenu.addItem(withTitle: "Quit", action:#selector(NSApplication.terminate), keyEquivalent: "q") 
}
    
func buildWnd() {
    
let _wndW : CGFloat = 400
let _wndH : CGFloat = 300

 window = NSWindow(contentRect:NSMakeRect(0,0,_wndW,_wndH),styleMask:[.titled, .closable, .miniaturizable, .resizable], backing:.buffered, defer:false)
 window.center()
 window.title = "Swift Test Window"
 window.makeKeyAndOrderFront(window)

// **** Button **** //
let myBtn = NSButton (frame:NSMakeRect( 100, 100, 175, 30 ))
 myBtn.bezelStyle = .rounded
 myBtn.autoresizingMask = [.maxXMargin,.minYMargin]
 myBtn.title = "Build Second Window"
 myBtn.action = #selector(self.myBtnAction(_:))
 window.contentView!.addSubview (myBtn)

// **** Quit btn **** //
let quitBtn = NSButton (frame:NSMakeRect( _wndW - 50, 10, 40, 40 ))
 quitBtn.bezelStyle = .circular
 quitBtn.autoresizingMask = [.minXMargin,.maxYMargin]
 quitBtn.title = "Q"
 quitBtn.action = #selector(NSApplication.terminate)
 window.contentView!.addSubview(quitBtn)
}
 
func applicationDidFinishLaunching(_ notification: Notification) {
 buildMenu()
 buildWnd()
}

func applicationShouldTerminateAfterLastWindowClosed(_ sender: NSApplication) -> Bool {
 return true
}

}
let appDelegate = AppDelegate()

// **** main.swift **** //
let app = NSApplication.shared
app.delegate = appDelegate
app.setActivationPolicy(.regular)
app.activate(ignoringOtherApps:true)
app.run()

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM