我有一个名为 LoginViewController
的 View Controller ,它符合 UITextFieldDelegate
并且有两个文本字段对象,即 Email
和 Password
。 Email
文本字段有一个电子邮件验证功能,如果验证失败,UIAlertController
实例将启动。
下面是我继续提问前的场景截图和代码:
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/