How do I get my UIPopoverController to animate its size when its contained UINavigationController pushes a new controller?
I have a UIPopover
being displayed from a UIBarButtonItem
in my iPad app. It contains a UINavigationViewController
, which has a sort of settings window as its root view controller. The settings window is a subclass of UITableViewController
(style set to grouped), and tapping any of its cells pushes different "chooser" view controllers on the nav controller that are also subclasses of UITableViewController
.
For each of the chooser views, in viewDidAppear
, I'm setting contentSizeForViewInPopover
appropriately:
self.contentSizeForViewInPopover = CGSizeMake(320, self.items.count * 44);
But it doesn't animate the change; when the navigation animation finishes, the popover snaps to the new height (width never changes from 320). Navigating backward animates the size change (accomplished with the technique from this answer), but forward never does.
I've tried obtaining a reference to the popover it's in and using setPopoverContentSize:animated:
but it doesn't work. I've looked at other questions to no avail.
How do I get it to always animate the size change properly?
Update: I've set up a simple test project to try this out. It's a tab bar application for iPad set up in Xcode. I added a tab bar item to the navigation bar in one of the view controllers. When that button is pressed, the controller presents a popover that contains a navigation controller that has a very simple UITableViewController
subclass, called TestContentViewController
, as its root view controller.
In viewDidLoad
of that subclass, I randomly generate a number of items:
self.numItems = arc4random() % 10 + 3;
This is my number of rows; number of sections is 1. In cellForRowAtIndexPath
I just set the cell's label text and return it. When a row is selected, I generate another instance of the same class and push it on the stack.
Without doing anything at all with the contentSizeForViewInPopover
property on any VC, the popover just goes to its maximum height and stays there no matter how many rows are in my table view.
If I set the size in viewDidAppear, like this:
-(void)viewDidAppear:(BOOL)animated
{
self.contentSizeForViewInPopover = CGSizeMake(320, self.numItems * 44);
[super viewDidAppear:animated];
{
- When the popover first appears, it flashes very quickly to full height and then snaps to the height I set.
- When a new controller is pushed on the navigation controller, it snaps to its height with no animation.
- When I navigate back, if the VC that I'm popping to is taller than the one I'm popping from, it animates to the correct size. If what I'm popping to is smaller, it does nothing.
If I do the same thing but in viewWillAppear
:
- When the popover first appears, it's full-height
- When I first tap a row and get a new controller on the stack, it animates to a minimum of about 400px tall. When a new controller is pushed on the stack, if it needs more height it gets it. If not it stays at what it was before.
If I do the same thing in viewDidLoad
, it's basically the same as viewWillAppear
except it appears at the right size at first.
I've tried setting the nav controller's delegate to be the VC that presents the popover, and then set the popover's height (setPopoverContentSize:animated:
) in navigationController:didShowViewController:animated:
, but the resultant height is off by a little bit. I think the size I set there needs to take into account the extra height of the navigation bar built into the top of the popover. And the animation when pushing a new controller on the stack is weird.
Update again: See here for the same problem solved with the newer UIPopoverPresentationController
.