How can I have a UIBarButtonItem with both image and text?
Asked Answered
D

6

45

When I try to use an image for a UIBarButtonItem, the text isn't shown. Is there a way to show both the text and the image?

Doering answered 11/10, 2010 at 1:35 Comment(0)
M
46

You can init the UIBarButtonItem with a custom view that has both image and text. Here's a sample that uses a UIButton.

UIImage *chatImage = [UIImage imageNamed:@"08-chat.png"];

UIButton *chatButton = [UIButton buttonWithType:UIButtonTypeCustom];
[chatButton setBackgroundImage:chatImage forState:UIControlStateNormal];
[chatButton setTitle:@"Chat" forState:UIControlStateNormal];
chatButton.frame = (CGRect) {
    .size.width = 100,
    .size.height = 30,
};

UIBarButtonItem *barButton= [[[UIBarButtonItem alloc] initWithCustomView:chatButton] autorelease];
self.toolbar.items = [NSArray arrayWithObject:barButton];
Minicam answered 11/10, 2010 at 3:26 Comment(7)
For the last line I get an error message saying "Request for member 'toolbar' is something not a structure or a union. How do I do it using self.navigationItem.rightBarButtonItem?Doering
In that case, replace the last line with self.navigationItem.rightBarButtonItem = barButton;. Or you can use this method: developer.apple.com/library/ios/documentation/uikit/reference/….Minicam
What happens if you use the setRightBarButtonItem:animated: method?Minicam
[self.navigationItem setRightBarButtonItem:barButton animated:YES];Minicam
Yes, that worked! Not what I expected, but it shows both text and graphics.Doering
It's not necessary to use a background image - you can just set the title and image of the UIButton and the image will automatically appear to the left of the text.Wilding
Bill:How to just set title and image?UIImage *image = [[UIImage imageNamed:@"Game.png"] imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal]; UIBarButtonItem *button = [[UIBarButtonItem alloc] initWithImage:image style:UIBarButtonItemStylePlain target:self action:@selector(refreshPropertyList:)]; button.title=@"3333"; It doesn't work.Natality
E
14

I suggest a way how to do this using Storyboard:

  1. Drag and drop to ToolBar usual UIView. It automatically will be wrapped to Bar Button Item. Adjust View width in "Size inspector". Change background color of View to clear color.

enter image description here

  1. Put UIButton and UILabel inside a View. You can use constraints to adjust their positions. Also you can put thier outside of a View, for example, little higher or less (like on my picture). Put Default and Highlighted images for UIButton.

  2. Copy Bar Button Item and adjust another buttons. Add Flexible spaces.

  3. Create outlets.

  4. Done :). When you run app, you get Tool Bar like this:

enter image description here

P.S. Here you can read how to create UIToolbar subclass using .xib

Edifice answered 10/1, 2016 at 16:6 Comment(2)
Nice solution. I made both the items UIButtons and put them in a vertical stack view.Brownout
Note to future self and other people struggling with this: if you are using a Navigation Controller Toolbar and are trying to do this in the "Toolbar Items" dropdown, make a dummy Toolbar and then copy the Bar Button Item over to Toolbar Items.Handlebar
D
7

Swift 4.2 with target

extension UIBarButtonItem {
    convenience init(image :UIImage, title :String, target: Any?, action: Selector?) {
        let button = UIButton(type: .custom)
        button.setImage(image, for: .normal)
        button.setTitle(title, for: .normal)
        button.frame = CGRect(x: 0, y: 0, width: image.size.width, height: image.size.height)

        if let target = target, let action = action {
            button.addTarget(target, action: action, for: .touchUpInside)
        }

        self.init(customView: button)
    }
}
Darcee answered 28/1, 2019 at 13:59 Comment(4)
it works. Any idea where the pregnant pauses come from between button press and arrival into the action specified?Autogiro
@AntonTropashko The problem is (1) the custom view makes it slower and (2) this is not the correct way to customize the Back button.Diatomaceous
Add this line to get rid of the left space: button.imageEdgeInsets = UIEdgeInsets(top: 0, left: -13.0, bottom: 0, right: 0)Hannahhannan
How do you then make the bar button text color match the icon?Jane
F
6
UIButton *button =  [UIButton buttonWithType:UIButtonTypeCustom];
[button setImage:[UIImage imageNamed:@"image.png"] forState:UIControlStateNormal];
[button addTarget:target action:@selector(buttonAction:)forControlEvents:UIControlEventTouchUpInside];
[button setFrame:CGRectMake(0, 0, 53, 31)];
UILabel *label = [[UILabel alloc]initWithFrame:CGRectMake(3, 5, 50, 20)];
[label setFont:[UIFont fontWithName:@"Arial-BoldMT" size:13]];
[label setText:title];
label.textAlignment = UITextAlignmentCenter;
[label setTextColor:[UIColor whiteColor]];
[label setBackgroundColor:[UIColor clearColor]];
[button addSubview:label];
UIBarButtonItem *barButton = [[UIBarButtonItem alloc] initWithCustomView:button];
self.navigationItem.leftBarButtonItem = barButton;
Falmouth answered 24/9, 2014 at 6:31 Comment(0)
M
2

Kris Markel's answer is a good start (only < iOS7), but if you use tint color, it wount work for the image and the highlighted state of the button.

I've created a category that will create a UIBarButtonItem (leftBarButtonItem) that looks and behaves like a normal iOS7 back button. Have a look at: https://github.com/Zero3nna/UIBarButtonItem-DefaultBackButton

Mullock answered 31/10, 2013 at 19:15 Comment(0)
V
1

Swift Update for UIBarButtonItem with a custom view that has both image and text.

var rightImage: UIImage  = UIImage(named: "YourImageName")
var rightButton : UIButton = UIButton(type: UIButtonType.custom)
rightButton.setBackgroundImage(rightImage, for: .normal)
rightButton.setTitle(title, for: .normal)
rightButton.frame.size = CGSize(width: 100, height: 30)
        
let menuUIBarButtonItem = UIBarButtonItem(customView: rightButton)
Vantassel answered 21/9, 2018 at 11:5 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.