I am using swift 4
and I am trying to create an alertView
when I there is an error while signing up a user using Firebase
. I have an IBAction
for the sign up button which will sign the user up using text from two textfields
, one for email and one for password.
I am basically trying to show an alertview
when there is an error with the sign up process, for example there is an empty textfield
.
I have attached a screenshot of the function to where that is occuring. I know that I am in fact getting an error because the print statement outputs an error if there is one.
Regardless of if there is an error or not, there is no alert view showing up and the app performs the segue regardless.
2019-01-15 21:40:26.368924-0500 Pronto[9036:225268] Warning: Attempt to present on whose view is not in the window hierarchy
This is the output that I am getting for the alertview now showing up. I have looked at all the other posts about this same issue but none seem to work.
This issue happens due to your view hierarchy.
You need to find out what is your current/Topmost view controller in view hierarchy and present your alert over it.
To find out topmost view controller use following code:
func getTopMostViewController() -> UIViewController? {
var topMostViewController = UIApplication.shared.keyWindow?.rootViewController
while let presentedViewController = topMostViewController?.presentedViewController {
topMostViewController = presentedViewController
}
return topMostViewController
}
And present your alert over topmost view controller and use main thread to present an alert because closures may have working on another thread.
DispatchQueue.main.async {
getTopMostViewController()?.present(alertController, animated: true, completion: nil)
}
Please refer to this stack answer: Swift 3 Attempt to present whose view is not in the window hierarchy
Try using ViewDidAppear
instead of View did Load.
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
let alertViewController = UIAlertController(title: "Any", message: "Any Custom Message", preferredStyle: .alert)
alertController.addAction(UIAlertAction(title: "ANy", style: .cancel, handler: nil))
present(alertViewController, animated: true, completion: nil)
}
You can get the top most view controller and have that view controller present the alert. So instead of self.present
use this approach and see if it works:
let topViewController = UIApplication.shared.keyWindow?.rootViewController
topViewController?.present(alertController, animated: true, completion: nil)
Also try presenting on the main thread, since you're trying to show the alert in createUser completion handler:
DispatchQueue.main.async {
self.present(alertController, animated: true, completion: nil)
}
Check if you have "double tap" issue:
- You double tap button accidentally
signUpBtnPressed
is called twice- First request is executed correctly, thus launching the segue, and the error is nil
- Second request returns error like 'user already exists', then trying to show alert from current controller, but segue is already launched and next controller is already presented
This is fixed by using loader with UI blocking (for example SVProgressHUD) - start loader at the beginning of the method and dismiss it in callback.
来源:https://stackoverflow.com/questions/54209766/swift-4-attempt-to-present-viewcontroller-whose-view-is-not-in-the-window-hierar