I have a UINavigationController
into which I push several views. Inside viewDidLoad
for one of these views I want to set the self.navigationItem.backBarButtonItem
to a custom view (based on a custom image). I don't know why, but it doesn't seem to work. Instead, I get the standard "back" button.
UIButton *backButton = [[UIButton alloc] initWithFrame:CGRectMake(0, 0, 63, 30)];
[backButton setImage:[UIImage imageNamed:@"back_OFF.png"] forState:UIControlStateNormal];
[backButton setImage:[UIImage imageNamed:@"back_ON.png"] forState:UIControlStateSelected];
UIBarButtonItem *backButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];
self.navigationItem.backBarButtonItem = backButtonItem;
[backButtonItem release];
[backButton release];
I tested with a standard title and it worked. What is wrong with the above code ?
self.navigationItem.backBarButtonItem = [[[UIBarButtonItem alloc] initWithTitle:@"Prout" style:UIBarButtonItemStyleDone target:nil action:nil] autorelease];
Thanks for any help on this.
It behaves strangely because setting a vc's backBarButtonItem doesn't change anything about the appearance of the vc's navigation item - instead, it changes the button that points BACK to the vc. See updating the navigation bar from Apple FMI.
That said I haven't had a whole lot of luck getting it to work myself. If you look around this site, you'll find some threads that suggest placing code very similar to what you already have immediately before the call to push a new view on the stack. I've had some luck there, but unfortunately not when it comes to using a custom image.
This is how I create a custom square back button with an arrow instead of the usual text.
I simply setup a delegate for my UINavigationController. I use the app delegate for that because the window root view controller is the UINavigationController i want to control.
So AppDelegate.m (ARC):
I'm using BlocksKit to catch the button tap event. It's very convenient for stuff like this but you can also use the regular addTarget:action:forControlEvents: method
As of iOS5 we have an excellent new way of customizing the appearance of almost any control using the UIAppearance protocol, i.e.
[UIBarButtonItem appearance]
. The appearance proxy allows you to create application wide changes to the look of controls. Below is an example of a custom back button created with the appearance proxy.Use the example code below to create a back button with custom images for normal and highlighted states. Call the following method from you appDelegate's
application:didFinishLaunchingWithOptions:
This is just a quick example. Normally you would want to have separate images for normal and highlighted (pressed) state.
If you are interested in customizing the appearance of other controls, some good examples can be found here: http://ios.biomsoft.com/2011/10/13/user-interface-customization-in-ios-5/
I too have been having problems with
customView
on anavigationItem.backBarButtonItem
. I suspect it's probably just b0rked in the SDK.While the workarounds outlined here do work, I've come up with a solution which is a little more elegant: it still uses
navigationItem.leftBarButtonItem
, but takes care of it for you automagically (no more need for 'child' view controllers to need to know anything about their 'parent' and/or to manually setnavigationItem.leftBarButtonItem
).First up, have some class be a
UINavigationControllerDelegate
for theUINavigationController
whose back button you're interested in. Then, in this class, set up something like the followingwillShowViewController
delegate method:I had some further problems with this; it seems that
UIBarButtonItem.action
andUIBarButtonItem.target
don't work when it's anavigationItem.leftBarButtonItem
. So, you're left with a custom back button that doesn't actually go back. I'll leave responding to touches in your custom view as an exercise for the reader (I used a UIButton), but you'll need add this method to your delegate class:and hook it up to fire when your custom view is tapped.
I think I found the solution for this. Simply set the button on the navigation item on the previous controller (The one that you want to go back to)
So if I have for example a root controller and I push a second controller and want to customize the back button then I should do the following:
Where self is the root view controller and not the second one.
I'm fairly certain that the
backBarButtonItem
is a read-only property. Instead of modifying thebackBarButtonItem
, try setting a customleftBarButtonItem
and hide thebackBarButtonItem
:You will also need to make sure you hook up the custom button to call the back action on the
UINavigationBar
.