Im calling popToRootViewController in a UiViewController that is acting as a sign-in page. The method is called once the backend has authenticated the user and will now allow the user to access their account. When the root view Controller is shown again, I get the following two errors.
nested push animation can result in corrupted navigation bar
Finishing up a navigation transition in an unexpected state. Navigation Bar subview tree might get corrupted.
Here is the code for signing in a user.
- (IBAction)signIn:(id)sender
{
NSString *userName = [self.emailTextField.text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
NSString *password = [self.passwordTextField.text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
if (userName.length == 0 || password.length ==0){
UIAlertView *alertView = [[UIAlertView alloc]initWithTitle:#"Invalid Login Credentials" message:#"Make sure you have entered a valid Username and Password" delegate:nil cancelButtonTitle:#"Ok" otherButtonTitles:nil];
[alertView show];
}else{
[PFUser logInWithUsernameInBackground:userName password:password block:^(PFUser *user, NSError *error) {
if (error){
UIAlertView *errorAlertView = [[UIAlertView alloc]initWithTitle:#"Something went wrong" message:[error.userInfo objectForKey:#"error"] delegate:nil cancelButtonTitle:#"ok" otherButtonTitles:nil];
[errorAlertView show];
}else{
//dispatch_sync(dispatch_get_main_queue(), ^{
[self.navigationController popViewControllerAnimated:YES];
//});
}
}];
}
}
The rootViewController is the initial view that the app launches into, and there it checks to see if there is a current user and if not the user will be sent to the login page. I do notice that by going into the Login page, there is a back arrow at the top of the navigation bar to go back to the rootViewController. Once The user signs in, there is now a back button in the same place that wants to take the user back to the LoginViewController which has been popped. I think this is where the issue is occurring.
This error is raised when you try to push several view controllers or pop more than one view controller. Are you sure the viewWillAppear method of your rootViewController is not attempting to push a view controller ? If so, you're attempting to push a view controller while another one is being popped. Hope this may help you!
Related
I have a view controller for registering a PFuser. I also have a view overlaid on the same view controller that occupies the entire frame but adjusted to be on the right side of the view controller.
This login view appears upon clicking "already registered" button the view controller. I also have the code to fill in the frame with view entirely.
The issue is: whenever I click on username textfield on the view to enter login credentials, it goes back to the view controller asking to register.
//"I am already registered" button action method
- (IBAction)registeredButton:(id)sender {
[UIView animateWithDuration:0.3 animations:^{
_loginOverlayView.frame = self.view.frame;
}];
}
//login button method
- (IBAction)loginButton:(id)sender {
[PFUser logInWithUsernameInBackground:_loginUsernameField.text password:_loginPasswordField.text block:^
(PFUser * user, NSError * error) {
if(!error)
{
NSLog(#"Login user");
[self performSegueWithIdentifier:#"login" sender:self];
_usernameField.text=nil;
_passwordField.text=nil;
_emailField.text=nil;
_reEnterPasswordField.text=nil;
_loginPasswordField.text=nil;
_loginUsernameField.text=nil;
}
else
{
UIAlertView *alert = [[UIAlertView alloc]initWithTitle:#"Oops!" message:#"Sorry we had problem logging you in!" delegate:self cancelButtonTitle:#"Ok" otherButtonTitles: nil];
[alert show];
}
}];
}
I would open a UIViewController from a UIAlertView button but it doesn't work.
When user terminate a game I show a UIAlertView with some text and two buttons: "Ok" to dismiss alert and "Score" to open the score page.
This is the actual UIAlertView:
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:#"YOU WIN!"
message:[NSString stringWithFormat:#"You win a game in %d seconds!",seconds]
delegate:self
cancelButtonTitle:#"Ok"
otherButtonTitles:#"Score", nil];
[alert show];
and this is the code for push the view:
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
{
NSString *title = [alertView buttonTitleAtIndex:buttonIndex];
if([title isEqualToString:#"Score"]){
ScoreViewController *score = [[ScoreViewController alloc]init];
[[self navigationController] pushViewController:score animated:YES];
}
}
What I obtain is a completely black screen. Can someone help me? I can't figure it out what I'm doing wrong.
PS: ScoreViewController has a segue from the root view but of course I can't create a new one from storyboard because I want to perform segue programmatically, from the alert view button.
Hope I've been clear, any help will be very appreciate!
It looks like you're instantiating an instance of ScoreViewController and pushing onto your navigation controller. While this is a legitimate way of presenting another controller, it's different from performing a Storyboard segue.
First, obviously, you'll have to make sure you have a segue connecting your view controllers in the Storyboard. It sounds like you've made it that far, so next you'll want to make sure the segue has an identifier. Select the segue and set its identifier in the Attributes inspector. Then perform the segue in code:
[self performSegueWithIdentifier:#"YourIdentifier" sender:self];
I got a button that leads to a tableViewController.
I created UIAlertView that gets triggered when the resultArray is null, which also means there is nothing to show on tableView but when user clicks ok, nothing changes and it goes to empty tableView.
How can I stop going into the other view controller?
this is my code
if (refilteredArray.count==0) {
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:#"No result"
message:#"Please enter a different combination."
delegate:nil
cancelButtonTitle:#"OK"
otherButtonTitles:nil];
[alert show];
}
The segue is not triggered with code, its a button triggered IBAction
- (IBAction)motorButton:(id)sender {
...
}
you can use following method
- (BOOL)shouldPerformSegueWithIdentifier:(NSString *)identifier sender:(id)sender
in the method body you can verify if your array is empty or not and if it is return NO. To be more specyfic you should check your segue identifier (remember to add it to your segue in IB first)
You need to go for if.. else block
if (refilteredArray.count==0) {
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:#"No result"
message:#"Please enter a different combination."
delegate:nil
cancelButtonTitle:#"OK"
otherButtonTitles:nil];
[alert show];
}else{
[self performSegueWithIdentifier:#"yourSegue" sender:sender];
}
For more info, check this link about segue
I'm new to iOS development and I'm using parse.com as my backend.
I want to navigate from [View Controller 2] to [view Controller 3]
but is not working right.
Here is the code
http://i.stack.imgur.com/t6zZg.jpg
NSString *username = [self.usernameField.text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
NSString *password = [self.passwordField.text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
NSString *email = [self.emailField.text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
if ([username length] == 0 || [password length] == 0 || [email length] == 0){
UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:#"Oops!" message:#"Make sure you enter a username, password , and email address!" delegate:nil cancelButtonTitle:#"OK" otherButtonTitles:nil];
[alertView show]; // This help to show your message successfully!
}
else {
PFUser *newUser = [PFUser user]; // This is a method to create new users
newUser.username = username;
newUser.password = password;
newUser.email = email;
[newUser signUpInBackgroundWithBlock:^(BOOL succeeded, NSError *error) {
if (error) {
UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:#"Sorry!" message:[error.userInfo objectForKey:#"error"] delegate:nil cancelButtonTitle:#"OK" otherButtonTitles: nil];
[alertView show];
}
else {
[self.navigationController popToViewController:_profilePhotoViewController animated:YES];
}
}];
}
}
According to the image you posted, VC3 is not displayed nor pushed to the controller before displaying VC2
This method popToViewController:animated: require that the controller is displayed before and in the controller stack in order to pop the stack back to it
You can achieve what you want by pop to root and then push your view
[self.navigationController popToRootViewControllerAnimated:NO];
[self.navigationController pushViewController:_profilePhotoViewController animated:YES];
Besure that _profilePhotoViewController is allocated before pushing it
.
.
UPDATE
In your storyboard set an identifier to your VC3 to be for example "VC3Identifier"
You can do this as you can see in the below image
Choose your controller
Show the utilities
show the identity inspector
Put 'VC3Identifier' in StoryBoard ID field
Then You will need to allocate a new instance of this controller and add it to the navigation controller
YourController *vc3 = [self.storyboard instantiateViewControllerWithIdentifier:#"VC3Identifier"];
[self.navigationController setViewControllers:#[vc3] animated:YES];
OK, well, I am not totally sure how your code works and what will trigger the change of View, but in your code where you call this:
[self.navigationController popToViewController:_profilePhotoViewController animated:YES];
Is what you need to handle (assuming profilePhotoViewController is VC3).
Go to your Storyboard and ctrl+click+drag from the presenting view (the VC that you are starting from) and end up at VC 3. VC 3 should light up blue. Let go and you will see a drop down menu. Select "Push". This will create a segue for you to transition. Next you need to give it an identifier, so click on the segue that was just made and go to the Attributes Inspector and in the "Identifier" field type in "ProfilePictureSegue" and then press enter (or return).
Then go to the file that contains the code you posted and replace
[self.navigationController popToViewController:_profilePhotoViewController animated:YES];
With:
[self performSegueWithIdentifier:#"ProfilePictureSegue"];
And bob is your uncle. Run it.
EDIT
I looked at your picture again and it looks like your VC3 is in the wrong place. Create a segue from VC2 to VC 3, not from the Navigation Controller to VC3. Call the segue, as I explained above, in the VC2 .m file, and it will transition between.
If you need more explanations.
I just learnt IOS programming and am writing my first iPhone app.
My app provides info on an MKMapView, the view controller of which is the root view controller of a UINavigationController. If the mobile signal is bad i use mapViewDidFailLoadingMap:withError: to make the app push one of two different view controllers onto the navigation controller stack depending on what the user is doing. Code follows:
- (void)mapViewDidFailLoadingMap:(MKMapView *)mapView withError:(NSError *)error
{
NSLog(#"mapViewDidFailLoadingMap: %#", [error localizedDescription]);
[aiView stopAnimating];
if (mapTypeInt == 0) {
NSString *message =
[NSString stringWithFormat:#"Your signal is not currently
strong enough to download a map. Switching to table view."];
UIAlertView *av = [[UIAlertView alloc] initWithTitle:#"Maps Unavailable"
message:message
delegate:nil
cancelButtonTitle:#"OK"
otherButtonTitles:nil];
[av show];
if (currentMode == #"retracingSteps")
{
RetraceViewController *rvc =
[[RetraceViewController alloc] initWithNibName:#"RetraceViewController" bundle:nil];
[[self navigationController] pushViewController:rvc animated:YES];
}
else{
TripTableViewController *ttvc = [[TripTableViewController alloc] init];
[[self navigationController] pushViewController:ttvc animated:YES];
}
}
else{
[self setMapType:0];
NSString *message = [NSString stringWithFormat:
#"Your signal is not currently strong enough to download a satellite map.
Switching to Standard Map view."];
UIAlertView *av = [[UIAlertView alloc] initWithTitle:#"Can't Use Satellite Maps"
message:message
delegate:nil
cancelButtonTitle:#"OK"
otherButtonTitles:nil];
[av show];
}
}
Yesterday I tested the poor mobile signal in a rural valley and the correct second view controller was pushed onto the stack. What I then noticed when I locked the phone and rechecked the app after a few minutes was that on waking, the root view controller was displayed quickly followed by the view controller I expected. In effect what this did was to push an identical copy of the second view controller onto the stack. I discovered this when I had to tap the back button half a dozen times to get back to the root view controller.
What I would like the app to do on wake up is to immediately display the view controller that was live when the phone is locked rather than the root view controller. I do not know how to do this.
This is probably happened because even you lock the screen your app still receiving locations update (location services can run code in a real background mode), also when you push a view controller the previous one in the stack still exist and still receiving location update and doing everything has been designed to do, so even you push another controller, if something happen that for the implemented logic has to push another controller, this view controller has access to the navigation controller so has just to add to the stack another controller and push it (the navigation controller still the same for all the VC in the stack)
So what you have to do is stop to handle this cases when you push another controller and restart to do that when you pop back the controller
This bug was caused by the handler mapViewDidFailLoadingMap:withError: being called multiple times before control passed to the pushed view controller. I fixed the bug by making three changes to the handler method:
1) I removed the UIAlertView show and moved it to the pushed view controller;
2) I used a flag (initialized in viewDidAppear:) to return from the handler without executing the push if the handler was being called a second time;
3) I checked for the pushed view controller's existence before allocing it.
When all three of these changes were made, transitions between view controllers happened correctly. Code follows:
- (void)mapViewDidFailLoadingMap:(MKMapView *)mapView withError:(NSError *)error
{
[aiView stopAnimating];
if (mapViewFailed > 0) {
mapViewFailed ++;
return;
}
if (mapTypeInt == 0) {
[map setUserTrackingMode:MKUserTrackingModeNone];
[self removeAsNoticationsObserver];
if ([currentMode isEqualToString:#"retracingSteps"])
{
if (!rvc) {
rvc = [[RetraceViewController alloc] initWithNibName:#"RetraceViewController" bundle:nil];
}
[[self navigationController] pushViewController:rvc animated:YES];
}
else{
if (!ttvc) {
ttvc = [[TripTableViewController alloc] init];
}
[[self navigationController] pushViewController:ttvc animated:YES];
}
mapViewFailed ++;
}
else{
[self setMapType:0];
NSString *message = [NSString stringWithFormat:#"Your signal is not currently strong enough to download a satellite map. Switching to Standard Map view."];
UIAlertView *av = [[UIAlertView alloc] initWithTitle:#"Can't Use Satellite Maps"
message:message
delegate:nil
cancelButtonTitle:#"OK"
otherButtonTitles:nil];
[av show];
}
}
No doubt a more elegant solution is available which I don't know about yet, being an iOS newb.