I am currently making an OS X application written in Swift. What I want to do is when the user enters text in a NSTextField
, I want to run a function that checks the value and adds it to a Label. How would I do this in swift?
- Conforms
ViewController
to protocolNSTextDelegate
. - Assign
ViewController
as Delegate forTextField
. Implement
controlTextDidChange
method.import Cocoa @NSApplicationMain class AppDelegate: NSObject, NSApplicationDelegate, NSTextFieldDelegate { @IBOutlet weak var window: NSWindow! @IBOutlet weak var textField: NSTextField! @IBOutlet weak var label: NSTextField! func applicationDidFinishLaunching(aNotification: NSNotification) { textField.delegate = self } override func controlTextDidChange(notification: NSNotification) { let object = notification.object as! NSTextField self.label.stringValue = object.stringValue } func applicationWillTerminate(aNotification: NSNotification) { // Insert code here to tear down your application } }
in ViewController:
import Cocoa
class ViewController: NSViewController, NSTextDelegate {
@IBOutlet weak var label: NSTextField!
@IBOutlet weak var label2: NSTextField!
@IBOutlet weak var textField: NSTextField!
@IBOutlet weak var textField2: NSTextField!
override func viewDidLoad() {
super.viewDidLoad()
}
override func controlTextDidChange(notification: NSNotification) {
if let txtFld = notification.object as? NSTextField {
switch txtFld.tag {
case 201:
self.label.stringValue = txtFld.stringValue
case 202:
self.label2.stringValue = txtFld.stringValue
default:
break
}
}
}
}
swift 5 xcode 10
//TODO: move oin a better place?
extension NSTextField{ func controlTextDidChange(obj: NSNotification){} }
class TracksController:NSViewController,
NSTextFieldDelegate
{
...
}
override func viewDidLoad() {
super.viewDidLoad()
//Search bar:
self.searchBar.delegate = self
NotificationCenter.default.addObserver(self,
selector: #selector(controlTextDidChange(_:)),
name: NSTextView.didChangeSelectionNotification,
object: nil)
}
//MARK: NSTextFieldDelegate
func controlTextDidChange(_ obj: Notification)
{
guard let object = obj.object as? NSTextView else{
return
}
let delegate = object.delegate
guard let searchField = delegate as? NSSearchField else {
return
}
let text = searchField.stringValue
print(text)
}
I know it’s been answered some while ago but I did eventually find this solution for macOS in Swift 3 (it doesn’t work for Swift 4 unfortunately) which notifies when a textfield is clicked inside and for each key stroke.
Add this delegate to your class:-
NSTextFieldDelegate
In viewDidLoad()
NotificationCenter.default.addObserver(self, selector: #selector(textDidChange(_:)), name: Notification.Name.NSTextViewDidChangeSelection, object: nil)
Then add this function:-
func textDidChange(_ notification: Notification) {
print("Its come here textDidChange")
guard (notification.object as? NSTextView) != nil else { return }
let numberOfCharatersInTextfield: Int = textFieldCell.accessibilityNumberOfCharacters()
print("numberOfCharatersInTextfield = \(numberOfCharatersInTextfield)")
}
Hope this helps others.
The answers above dont work for swift 4 so,
Solution for Swift 4
The process is mostly the same but you just have to extend some functionality to the NSTextfield Class. The reason we have to do this is that the delegate protocol methods like controlTextDidChange: actually don't exist at all because of the old design of objective-c code that doesn't map well to swift. The NSTextFieldDelegate in objective-c is an informal protocol which is the same thing as an extension in swift.
Example
extension NSTextField{ func controlTextDidChange(obj: NSNotification){} }
class SomeViewController:NSViewController,NSTextFieldDelegate {
override func controlTextDidChange(_ obj: Notification)
{
let object = obj.object as! NSTextField
let value = object.stringValue
print(value)
}
For every function, you want to use from the NSTextFieldDelegate just put the method in the extension clause.
//If you want to trim Ex: Email ID, please use the below code.
class ViewController: NSViewController {
@IBOutlet weak var emailTextField: NSTextField!
override func viewDidLoad() {
super.viewDidLoad()
emailTextField.delegate = self
}
}
extension ViewController : NSTextFieldDelegate {
func controlTextDidBeginEditing(_ obj: Notification) {
print(#function)
}
func controlTextDidEndEditing(_ obj: Notification) {
print(#function)
}
func controlTextDidChange(_ obj: Notification) {
print(#function)
if let textField = obj.object as? NSTextField {
let givenText = textField.stringValue
//Here you add other types of white space
let possibleWhiteSpace: NSArray = [" ", "\t", "\n\r", "\n","\r"]
var string: String = givenText as String
possibleWhiteSpace.enumerateObjects { (whiteSpace, idx, stop) -> Void in
string = string.replacingOccurrences(of: whiteSpace as! String, with: "") as String
}
print(string)
emailTextField.stringValue = string as String
}
}
}
extension String {
func removeAllTypesOfWhiteSpaces(_ inputString : String) -> String {
//Here you add other types of white space
let possibleWhiteSpace: NSArray = [" ","\t", "\n\r", "\n","\r"]
var outputStr: String = inputString as String
possibleWhiteSpace.enumerateObjects { (whiteSpace, idx, stop) -> Void in
outputStr = outputStr.replacingOccurrences(of: whiteSpace as! String, with: "") as String
}
return outputStr
}
}
extension LoginViewController : NSTextFieldDelegate {
func controlTextDidChange(_ obj: Notification) {
if let textField = obj.object as? NSTextField {
let givenText = textField.stringValue
let resultStr = givenText.removeAllTypesOfWhiteSpaces(givenText)
textField.stringValue = resultStr as String
}
}
}
© 2022 - 2024 — McMap. All rights reserved.