ios - 执行 UIAlertController 实例时,在单个文本字段对象上多次触发 textFieldShouldEndEditing 委托(delegate)方法

标签 ios swift xcode uialertcontroller uitextfielddelegate

我有一个名为 LoginViewController 的 View Controller ,它符合 UITextFieldDelegate 并且有两个文本字段对象,即 EmailPasswordEmail 文本字段有一个电子邮件验证功能,如果验证失败,UIAlertController 实例将启动。

下面是我继续提问前的场景截图和代码:

Log In Scene

Log In Scene with Error

ps: 有很多打印语句让我理解执行流程

func textFieldShouldEndEditing(textField: UITextField) -> Bool {
    guard self.validateEmail(self.emailTextField.text!) else {
        print("self.presentedViewController: \(self.presentedViewController)")
        if self.presentedViewController == nil {
            let alertController = UIAlertController(title: "Invalid Email Address", message: "Please try again.", preferredStyle: UIAlertControllerStyle.Alert)
            alertController.addAction(UIAlertAction(title: "OK", style: .Default, handler: nil))
            self.presentViewController(alertController, animated: true, completion: nil)
        }
        print("textFieldShouldEndEditing returnin false: \(textField.placeholder!) text field w/ value: \(textField.text!)")
        return false
    }
    print("textFieldShouldEndEditing returning true: \(textField.placeholder!) text field w/ value: \(textField.text!)")
    return true
}

func textFieldShouldBeginEditing(textField: UITextField) -> Bool {
    print("textFieldShouldBeginEditing: \(textField.placeholder!) text view")
    return true
}

func textFieldDidEndEditing(textField: UITextField) {
    print("textFieldDidEndEditing: \(textField.placeholder!) text field with value: \(textField.text!)")
}

func textFieldDidBeginEditing(textField: UITextField) {
    print("textFieldDidBeginEditing: \(textField.placeholder!) text field")
}

func textFieldShouldReturn(textField: UITextField) -> Bool {
    if !(self.emailTextField.text?.isEmpty)! && !(self.passwordTextField.text?.isEmpty)! {
        self.emailTextField.resignFirstResponder()
        self.passwordTextField.resignFirstResponder()
    }
    else if !(self.emailTextField.text?.isEmpty)! {
        self.passwordTextField.becomeFirstResponder()
    }
    print("textFieldShouldReturn: \(textField.placeholder!) text field w/ value: \(textField.text!)")
    return true
}

当在 Email 文本字段中输入格式正确的电子邮件地址并且通过下一个键或手动点击将焦点移至下一个文本字段(密码)下一个文本字段 (Password),以下委托(delegate)方法链按预期(和理解)发生:

控制台输出:

格式:[委托(delegate)方法名称:文本字段占位符值(+文本值)]

 textFieldShouldBeginEditing: Email text field
 textFieldDidBeginEditing: Email text field
 textFieldShouldBeginEditing: Password text field
 textFieldShouldEndEditing returning true: Email text field w/ value: abc@abc.co
 textFieldDidEndEditing: Email text field w/ value: abc@abc.co
 textFieldDidBeginEditing: Password text field
 email field is filled
 textFieldShouldReturn: Email text field w/ value: abc@abc.co

当在 Email 文本字段中检测到格式不正确的电子邮件地址时,就会出现此问题。 UITextFieldDelegate 方法 textFieldShouldEndEditing 被调用 三次 而不是一次作为 UIAlertController 的实例出现。

ps:警告 View 只在屏幕上显示一次。

控制台输出: 格式:[委托(delegate)方法名称:文本字段占位符值(+文本值)]

 textFieldShouldBeginEditing: Email text field
 textFieldDidBeginEditing: Email text field
 textFieldShouldBeginEditing: Password text field

 //1st occurence
 self.presentedViewController: nil
 textFieldShouldEndEditing returnin false: Email text field w/ value: bob

 //2nd occurence
 self.presentedViewController: Optional(<UIAlertController: 0x7f9018662df0>)
 textFieldShouldEndEditing returnin false: Email text field w/ value: bob

 //3rd occurence
 self.presentedViewController: Optional(<UIAlertController: 0x7f9018662df0>)
 textFieldShouldEndEditing returnin false: Email text field w/ value: bob

基于这个 so 的线程的解释:"textFieldShouldEndEditing called multiple times" ,我的理解是,当焦点即将离开 Email 文本字段时(textFieldShouldEndEditing 方法的第一次出现),焦点已经Password 文本字段上,从而获得第一响应者状态。但是,由于检测到无效电子邮件,会出现一个警报 Controller ,Password 文本字段突然失去焦点,因此也会触发 textFieldShouldEndEditing。所以,我预计 textFieldShouldEndEditing 只会被触发两次(一次是由于 Email 文本字段。另一个是由于 Password 文本字段)而不是像这样的三次(都是由于 Email 文本字段):

 textFieldShouldBeginEditing: Email text field
 textFieldDidBeginEditing: Email text field
 textFieldShouldBeginEditing: Password text field

 self.presentedViewController: nil
 textFieldShouldEndEditing returnin false: Email text field w/ value: bob

 self.presentedViewController: Optional(<UIAlertController: 0x7f9018662df0>)
 textFieldShouldEndEditing returnin false: Password text field w/ value: nil

但是,事实并非如此。任何人都可以启发为什么我的理解不正确。感谢您提供的任何指针。

最佳答案

将您的电子邮件验证码放入

func textFieldDidEndEditing(textField: UITextField)

代替

func textFieldShouldEndEditing(textField: UITextField) -> Bool

在显示警报 Controller 之前,您还可以添加(如果需要)

self.view.endEditing(true)

然后点击确定您可以设置的操作

self.emailTextField.becomeFirstResponder()

希望这种方法能解决您的问题。

关于ios - 执行 UIAlertController 实例时,在单个文本字段对象上多次触发 textFieldShouldEndEditing 委托(delegate)方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38844280/

相关文章:

iphone - 我可以在 iPad 模拟器中仅输入授权用户时检查登录屏幕吗?

iOS - 如何启用与 WebView 的用户交互

ios - 如何在 TableViewController 中显示 Firestore 文档名称?

ios - 使用 SQLITE 的 Swift 应用程序在更新数据库后崩溃。 SQLITE 文件未复制到文件/包

ios - 数组中最后添加的对象

ios - 如何在iOS应用程序中转到 "Home-page"?

ios - 如何将@noescape 注释添加到可选闭包

ios - "Expression was too complex to be solved in reasonable time..."是什么意思,我该如何解决?

ios - 使对象/图像在 Xcode 中随机出现和消失

ios - 您如何快速访问用户联系人的电话号码?