4

I have a UINavigationController and I have seperate UIViews that I switch between using a UISegmentControl. On switching the views, I add the view as a subview to my navigation controller's view:

[self.view addSubview:segmentTab1.view];

and

[self.view addSubview:segmentTab2.view];

Then, in the subViews, each has a UITableView, but my issue is, that I am unable to push a new viewController into view in the didSelectRowAtIndexPath method.

The method is called correctly and by setting breakpoints, I can see the method for pushing the view gets called as well, but nothing happens. This is my code for pushing it:

[self.navigationController pushViewController:detailsViewController animated:YES];

I also tried

[super.navigationController pushViewController:detailsViewController animated:YES];

What am I doing wrong - or is is just not possible to do it with a subview?

runmad
  • 14,846
  • 9
  • 99
  • 140

6 Answers6

6

When you call -pushViewController, which view controller is self? If you are calling that from within one of your tab subviews, self likely doesn't have a reference to the navigation controller from the top level view that you added it to. You can verify this by getting the memory address of the navigation controller from within the top level view and comparing it to what you have in the subview. Obviously if it's nil or doesn't match, then that's your problem.

You only get a navigation controller "for free" when it's been added to the navigation stack itself with -pushViewController which your subviews haven't been added that way.

Matt Long
  • 24,438
  • 4
  • 73
  • 99
  • Exactly -- the call to pushViewController must come from a view that has itself been pushed onto the navigation controller. It does not necessarily have to be the last one pushed, but it DOES have to be on the navigation controllers stack. – Amagrammer Nov 06 '09 at 17:39
  • 1
    Clarification: "must come from a view CONTROLLER" -- not a view. – Amagrammer Nov 06 '09 at 17:40
  • Thanks, Matt. And 'Amagrammer'. I have tabs, with NavigationControllers in each, which have a nib and a class attached to their views. I am going to set it up, so each tab has a view controller, and in each of the viewcontroller, I add a UINavigationController, which switches out it's views based on the segmentedControl and hopefully I can push in subviews, when I get there :) – runmad Nov 06 '09 at 18:02
  • Basically. The App Store app on the iPhone is an example of what I want to achieve. TabBar, each with a UINavigationController that have segmentedController (some of them do at least) and the rows can be pushed. – runmad Nov 06 '09 at 18:12
  • I agree, when you add a subview to an existing view, it won't include a reference to the navigation controller and your pushViewController:animated: calls will be sent to nil and ignored. Interestingly, the App Store application (and several others) look like they use segmented controls to filter table data in a single nib, rather than as a selection between different nibs (essentially a tab view). – crafterm Apr 21 '10 at 06:27
  • But can you explain me in detail how to call a different screen from subview. I have this exact problem. – dinesh Jan 13 '12 at 09:46
2

I had a similar issue when implementing a common header for all the views After many tries , i have fixed it by this -

In all the viewController

- (void)viewWillAppear:(BOOL)animated {

    [self.view addSubview:[[(NavAppAppDelegate *)[[UIApplication sharedApplication] delegate] headerview] view]];
    [[(NavAppAppDelegate *)[[UIApplication sharedApplication] delegate] headerview] viewWillAppear:YES];

}

I have referred following post to implement the header view Common XIB in multiple View Controller in iPhone

[self.view addSubview:[[(NavAppAppDelegate *)[[UIApplication sharedApplication] delegate] headerview] view]];     // line will load the header subview 

[[(NavAppAppDelegate *)[[UIApplication sharedApplication] delegate] headerview] viewWillAppear:YES]; // this is to call the viewWillAppear method of HeaderController where we can write code to check the user is logged or not and change the login button to logout button etc ..

Community
  • 1
  • 1
Rijesh
  • 31
  • 1
  • 3
1

Instead of

[self.view addSubview:segmentTab1.view];

and

[self.view addSubview:segmentTab2.view];

you may use

[self pushViewController: segmentTab1 animated: NO];

and

[self pushViewController: segmentTab2 animated: NO];

to add your viewControllers to the navigation hierarchy, and make [super.navigationController pushViewController:detailsViewController animated:YES]; work.

luvieere
  • 37,065
  • 18
  • 127
  • 179
  • Not sure if this will work, but if you switch out views in the segmentController, and you use a TabBar, tapping the tabBarItem for that view will push back the view. Doesn't look so good :( – runmad Nov 06 '09 at 18:07
  • I'm not sure I understand this behavior correctly. Perhaps you should look at *visibleViewController* property of the navigationController, and if it corresponds to the view you're trying to push, don't push it anymore. – luvieere Nov 06 '09 at 18:22
0

It is not entirely clear to me what your view hierarchy is, but in general if your navigation controllers view is not the first subview of a window or an element of one of Apple's collection views (either another navigation view controller's content view or a tab controller's content view) it won't work correctly.

Louis Gerbarg
  • 43,356
  • 8
  • 80
  • 90
0

One possibility, if you are not averse to singletons, is to make your product's UINavigationController object be a singleton, accessible from (for example) your application delegate.

You would invoke it thus:

[[((MyApplicationDelegate*)[UIApplication delegate]) navController] 
                pushViewController: viewControllerToPush animated: YES];

where within MyApplicationDelegate, navController returns the singleton object.

Amagrammer
  • 6,385
  • 3
  • 28
  • 30
0
LogEntryDetailViewController *leController = [[LogEntryDetailViewController alloc] initWithNibName:@"LogEntryDetailView" bundle:[NSBundle mainBundle]];
    [leController setTitle:[NSString stringWithFormat:@"%@", [logEntriesArray objectAtIndex:row]]];
    [[self navigationController] pushViewController:leController animated:NO];      
    [leController release], leController = nil;
dreadbot
  • 942
  • 2
  • 11
  • 28