I have added a UITextField to UIAlertController, but shouldChangeCharactersInRange will get not fired. Why? I set the delegate.
let alertController = UIAlertController(title: "", message: "xxx", preferredStyle: .Alert)
self.presentViewController(alertController, animated:true, completion:nil)
let textField = UITextField()
textField.delegate = self
alertController.addTextFieldWithConfigurationHandler(nil)
and in the same class, the delegate:
func textField(textField: UITextField!,
    shouldChangeCharactersInRange range: NSRange,
    replacementString string: String!) -> Bool {
The text field that you're setting the delegate for is not the same text field that is added to the alert controller. Basically, you're creating a new instance of UITextField, but never giving it a frame, or adding it to the view hierarchy. At the same time, you're using addTextFieldWithConfigurationHandler() to add a text field to the alert controller, but you never set the delegate for this text field. I believe this is what you want:
let alertController = UIAlertController(title: "", message: "xxx", preferredStyle: .Alert)
alertController.addTextFieldWithConfigurationHandler {[weak self] (textField: UITextField!) in
    textField.delegate = self
}
self.presentViewController(alertController, animated:true, completion:nil)
I was unable to get this working with UITextFieldDelegate. The delegate was set correctly but not being called for the UITextField within a UIAlertController.
Based on the answer here How do I validate TextFields in an UIAlertController? I learned you can instead use addTarget for UIControl.Event.editingChanged to call a selector when the editing changes.
let alertController = UIAlertController(title: "Title", message: "message", preferredStyle: .alert)
alertController.addTextField { (textField : UITextField!) -> Void in
    /*
     * Alternative to UITextFieldDelegate
     */
    textField.addTarget(alertController, action: #selector(alertController.textDidChange), for: .editingChanged)
}
let searchAction = UIAlertAction(title: "Search", style: .default)
let cancelAction = UIAlertAction(title: "Cancel", style: .cancel, handler: nil )
alertController.addAction(searchAction)
alertController.addAction(cancelAction)
present(searchAlertController, animated: true)
You can extend or subclass UIAlertController to add the selector:
extension UIAlertController {
    @objc func textDidChange() {
        guard let textField = textFields?.first else { return }
        guard let searchAction = actions.first(where: { $0.title == "Search" }) else { return }
        let text = textField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
        searchAction.isEnabled = !text.isEmpty
    }
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With