ChatGPT解决这个技术问题 Extra ChatGPT

iOS7 UIModalTransitionStyleFlipHorizontal bounces after transition

I'm updating my app for iOS 7 and I discovered a weird problem. I'm presenting a UIViewController wrapped in a UINavigationController with UIModalTransitionStyleFlipHorizontal.

In iOS 6 it works fine, but in iOS 7 the navigation bar bounces after the transition. Does this have something to do with the status bar? I've set translucency of the main navigation bar to NO.

In the Info.plist, View controller-based status bar appearance is set to NO.

And here is a GIF showing the problem in a minimal demo app:

https://i.stack.imgur.com/EznOv.gif

Here is my code:

feedNavigationController = [[UINavigationController alloc] init];
feedNavigationController.navigationBar.translucent = NO;

SettingsViewController *settingsVC = [[SettingsViewController alloc] init];

feedNavigationController.modalTransitionStyle = UIModalTransitionStyleFlipHorizontal;
[feedNavigationController setViewControllers:[NSArray arrayWithObjects:settingsVC, nil]];

[self presentViewController:feedNavigationController animated:YES completion:nil];
Unfortunately the video cannot be found.
I have the same issue.
Same issue exactly, the GM build has some Biig issues, collectionView reloadData is one too!
Not for me! iOS 8 still has this problem. self.navigationController?.navigationBar.layer.removeAllAnimations() in viewwillappear fixed it for me.
Same in my project, on iOS9 I use UIView transitionFromView:toView:duration:options:completion: to switch between two viewcontrollers and facing this bug. Following answers solved problem with navigation bar height, but not with it's tint color, anyone faced this problem?

B
Ben Packard

This appears to be a UIKit bug. The following workaround seems to resolve the issue for me.

- (void)viewWillAppear:(BOOL)animated
{
    [super viewWillAppear:animated];

    [self.navigationController.navigationBar.layer removeAllAnimations];
}

(Place this in the view controller you are transitioning to).


This worked for me. Make sure to do it in the root view controller of your navigation controller, not the navigation controller itself.
Interesting - what exactly happens on dismissal?
The same thing happens on dismissal. So this only partially solves the problem. Adding this to the view controller that began the transition (the from view controller) doesn't fix the issue when dismissing and transitioning back.
I see - I flip from a controller without a navbar, so never encountered this. Technically it resolves the stated question ;)
This worked for me the first time the animation is called, but not in any subsequent calls. Why would this be, since it's called every time the view will appear??? Maddening.
m
mluisbrown

To solve this problem for present & dismiss, I use the iOS7 custom transition.

Add this to your UIViewController :

- (id <UIViewControllerAnimatedTransitioning>)animationControllerForPresentedController:(UIViewController *)presented presentingController:(UIViewController *)presenting sourceController:(UIViewController *)source {
    return (id<UIViewControllerAnimatedTransitioning>)self;
}

- (id <UIViewControllerAnimatedTransitioning>)animationControllerForDismissedController:(UIViewController *)dismissed {
    return (id<UIViewControllerAnimatedTransitioning>)self;
}

- (NSTimeInterval)transitionDuration:(id <UIViewControllerContextTransitioning>)transitionContext {
    return 0.7f;
}

- (void)animateTransition:(id <UIViewControllerContextTransitioning>)transitionContext {
    UIView *containerView = [transitionContext containerView];


    UIViewController *fromVC = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
    [containerView addSubview:fromVC.view];

    UIViewController *toVC = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];
    [containerView addSubview:toVC.view];

    UIViewAnimationOptions animationOption = ([toVC.presentedViewController isEqual:fromVC])?UIViewAnimationOptionTransitionFlipFromLeft:UIViewAnimationOptionTransitionFlipFromRight;


    [UIView transitionFromView:fromVC.view
                        toView:toVC.view
                      duration:[self transitionDuration:transitionContext]
                       options:animationOption
                    completion:^(BOOL finished) {
                        [transitionContext completeTransition:YES];
                    }];
}

To use it, you just had to check if you are on iOS7 and set the transitionDelegate :

YourVCWithTheCustomTransition* yourVC = [[YourVCWithTheCustomTransition alloc] init];

CGFloat deviceVersion = [UIDevice currentDevice].systemVersion.floatValue;
if(deviceVersion >= 7.0) [yourVC setTransitioningDelegate:yourVC];

[self presentModalViewController:yourVC animated:YES];
[yourVC release];

In my case, I had a custom UINavigationController where the custom transition is defined : i don't have to do this each time.


This worked beautifully with the addition of "else yourVC.modalTransitionStyle = UIModalTransitionStyleFlipHorizontal;" in the test of deviceVersion so it continues to work in iOS6.
This should be the accepted answer. Works perfectly. My only suggestion would be to put this code in a separate class which you can then use in any View Controller.
Don't check for the system version but check whether the method is availbe: if ([yourVC respondsToSelector:@selector(setTransitioningDelegate:)] { [yourVC setTransitioningDelegate:yourVC]; }
Great it works, only additional thing I had to do was set the delegate in 'viewwillappear' since some of my views have segues defined via the storyboard editor.
Funny enough, returning from this transition can cause a permanent white screen under iOS 8. Of course you don't need to do any of this under iOS 8 because the bug that made this necessary was fixed.
R
Rishil Patel

This appears to be a UIKit bug. The following workaround seems to resolve the issue for me.

presentViewController (place this in the view controller you are transitioning to):

- (void)viewWillAppear:(BOOL)animated
{
    [super viewWillAppear:animated];

    [self.navigationController.navigationBar.layer removeAllAnimations];
}

dismissViewControllerAnimated (place this in the view controller you dismiss to):

- (void)viewWillLayoutSubviews{
    [super viewWillLayoutSubviews];

    [self.navigationController.navigationBar.layer removeAllAnimations];
}

if you don't use autolayout. you need add this to the view controller you dismiss to:

- (void)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];

    [self.view setNeedsLayout];
} 

sorry, I paste wrong code last time. This code worked for me.
This definitely worked in my case. Storyboard screenshot: cl.ly/image/060t10100b2W
I can confirm this worked for me as well. iOS 7.1.1
R
Rishil Patel

I had the same issue and could "solve it" (it's not a real solution to the problem but it looks fine :) ). The trick is present the view controller using pushViewController/popViewController with an UIView animation to make a flip. Here is a example code to present the view controller:

UIViewController *viewController = [[UIViewController alloc] init];
[UIView transitionWithView:self.navigationController.view 
                  duration:0.5 
                   options:UIViewAnimationOptionTransitionFlipFromLeft 
                animations:^{
                   [self.navigationController pushViewController:viewController animated:NO];
                }
                completion:nil];

To dismiss it:

[UIView transitionWithView:self.navigationController.view 
                  duration:0.5 
                   options:UIViewAnimationOptionTransitionFlipFromRight 
                animations:^{
                   [self.navigationController popViewControllerAnimated:NO];
                }
                completion:nil];

If you don't want the navigationBar on the pushed controller just call [self.navigationController setNavigationBarHidden:YES animated:NO] in viewWillAppear. I hope this approach help you.


This actually solves the problem of both sides of the transition, which is not something the above answer does. If you have a navbar on the modal controller, you might need to set the left bar button item to nil to get the exact effect you had on iOS6(no back button) but that's about it.
By the way, for some reason, this method results in visual glitches in the navbar(though not as severe as not using it). Mainly, the title and back button text don't show up until after the transition and are blank during.
This should be the accepted answer, it's a perfect workaround for transitioning in both directions. just check hides bottom bar on push and hide/unhide navigation bar as needed in the view will appear method of the pushing and pushed controller.
O
Ortwin Gentz

For both the presenting and the presented view controller I have a UITableViewController within UINavigationController, both configured using Auto Layout. I noticed that the other answers didn't solve the problem that on dismiss the tableView of the presenting view controller jumps 20 pt vertically.

This solution addresses this problem.

In the presented view controller (as proposed by Ben Packard):

- (void)viewWillAppear:(BOOL)animated {
    [super viewWillAppear:animated];
    [self.navigationController.navigationBar.layer removeAllAnimations];
}

In the presenting view controller (as proposed in part by dusty):

- (void)viewWillLayoutSubviews{
    if (self.navigationController.presentedViewController) {
        [self.navigationController.navigationBar.layer removeAllAnimations];
        [self.tableView.layer removeAllAnimations];
    }
    [super viewWillLayoutSubviews];
}

t
titicaca

The same for me. What actually worked is to change style to CoverVertical, looks much smoother.


关注公众号,不定期副业成功案例分享
Follow WeChat

Success story sharing

Want to stay one step ahead of the latest teleworks?

Subscribe Now