Editable NSTextField inside not .titled NSPanel

159 Views Asked by At

I'm making a bridge for react-native-macos. I need NSPanel with the same behaviour as Spotlight.

I programmatically created NSPanel and NSTextField inside. Everything working as expected, but if I change NSPanel to not .titled - the text field is disabled.

Works:

panel = NSPanel(contentRect: NSRect(x: 0, y: 0, width: 400, height: 40), styleMask: [
    .borderless,
    .nonactivatingPanel,
    .titled,  < ------- HERE
    .resizable,
  ], backing: .buffered, defer: true)

searchField = NSTextField()
searchField.delegate = self
searchField.isBezeled = false
searchField.font = NSFont.systemFont(ofSize: 20, weight: .light
searchField.drawsBackground = false
searchField.placeholderString = "Query here..."
searchField.setFrameSize(NSMakeSize(400, 40)

Doesn't work:

panel = NSPanel(contentRect: NSRect(x: 0, y: 0, width: 400, height: 40), styleMask: [
    .borderless,
    .nonactivatingPanel,
    .resizable,
  ], backing: .buffered, defer: true)

searchField = NSTextField()
searchField.delegate = self
searchField.isBezeled = false
searchField.font = NSFont.systemFont(ofSize: 20, weight: .light
searchField.drawsBackground = false
searchField.placeholderString = "Query here..."
searchField.setFrameSize(NSMakeSize(400, 40)

How can I do NSPanel with hidden titlebar and editable NSTextField inside?

2

There are 2 best solutions below

0
On BEST ANSWER

I believe .borderless and .titled are mutually exclusive. In other words, when you specify .borderless in the top code, the .titled overrides it and you get end up getting a "normal window" that handles events normally. In the second code, the .borderless actually registers and you end up getting a borderless window subclass that handles events quite differently. But all of this is beside the point.

Assuming you can require greater than macOS 10.10, you can use the following code:

panel = NSPanel(contentRect: NSRect(x: 0, y: 0, width: 400, height: 40), styleMask: [
    .titled,
    .fullSizeContentView,
    ], backing: .buffered, defer: true)
panel.titleVisibility = .hidden
panel.titlebarAppearsTransparent = true
panel.makeFirstResponder(searchField)
panel.makeKeyAndOrderFront(nil)

This gets a full-sized titled panel that handles events normally (it allows its views to become key). Then before showing the panel, hide the titlebar and also make it transparent. Note that to have the titlebar fully disappeared, you need to not use any of the .resizable, .closable, or .miniaturizable options.

0
On

Simply adding this in my class which inherits from NSPanel and represents a floating panel helped and made the text field responsive.

override var canBecomeKey: Bool {
    return true
}

override var canBecomeMain: Bool {
    return true
}

Found in this piece of code: https://gist.github.com/jordibruin/8ae7b79a1c0ce2c355139f29990d5702