Remove tab bar item text, show only image

IosSwiftUitabbaritem

Ios Problem Overview


Simple question, how can I remove the tab bar item text and show only the image?

I want the bar items to like in the instagram app:

enter image description here

In the inspector in xcode 6 I remove the title and choose a @2x (50px) and a @3x (75px) image. However the image does not use the free space of the removed text. Any ideas how to achieve the same tab bar item image like in the instagram app?

Ios Solutions


Solution 1 - Ios

You should play with imageInsets property of UITabBarItem. Here is sample code:

let tabBarItem = UITabBarItem(title: nil, image: UIImage(named: "more")
tabBarItem.imageInsets = UIEdgeInsets(top: 9, left: 0, bottom: -9, right: 0)

Values inside UIEdgeInsets depend on your image size. Here is the result of that code in my app:

enter image description here

Solution 2 - Ios

// Remove the titles and adjust the inset to account for missing title
for(UITabBarItem * tabBarItem in self.tabBar.items){
    tabBarItem.title = @"";
    tabBarItem.imageInsets = UIEdgeInsetsMake(6, 0, -6, 0);
}

Solution 3 - Ios

Here is how you do it in a storyboard.

Clear the title text, and set the image inset like the screenshot below

enter image description here

Remember the icon size should follow the apple design guideline

enter image description here

This means you should have 25px x 25px for @1x, 50px x 50px for @2x, 75px x 75px for @3x

Solution 4 - Ios

Using approach with setting each UITabBarItems title property to "" and update imageInsets won't work properly if in view controller self.title is set. For example if self.viewControllers of UITabBarController are embedded in UINavigationController and you need title to be displayed on navigation bar. In this case set UINavigationItems title directly using self.navigationItem.title, not self.title.

Solution 5 - Ios

If you're using storyboards this would be you best option. It loops through all of the tab bar items and for each one it sets the title to nothing and makes the image full screen. (You must have added an image in the storyboard)

for tabBarItem in tabBar.items!
{
   tabBarItem.title = ""
   tabBarItem.imageInsets = UIEdgeInsetsMake(6, 0, -6, 0)
}

Solution 6 - Ios

Swift version of ddiego answer

Compatible with iOS 11

Call this function in viewDidLoad of every first child of the viewControllers after setting title of the viewController

Best Practice:

Alternativelly as @daspianist suggested in comments

> Make a subclass of like this class BaseTabBarController: > UITabBarController, UITabBarControllerDelegate and put this function > in the subclass's viewDidLoad

func removeTabbarItemsText() {

    var offset: CGFloat = 6.0

    if #available(iOS 11.0, *), traitCollection.horizontalSizeClass == .regular {
        offset = 0.0
    }

    if let items = tabBar.items {
        for item in items {
            item.title = ""
            item.imageInsets = UIEdgeInsets(top: offset, left: 0, bottom: -offset, right: 0)
        }
    }
}

Solution 7 - Ios

iOS 11 throws a kink in many of these solutions, so I just fixed my issues on iOS 11 by subclassing UITabBar and overriding layoutSubviews.

class MainTabBar: UITabBar {

    override func layoutSubviews() {
        super.layoutSubviews()
    
        // iOS 11: puts the titles to the right of image for horizontal size class regular. Only want offset when compact.
        // iOS 9 & 10: always puts titles under the image. Always want offset.
        var verticalOffset: CGFloat = 6.0

        if #available(iOS 11.0, *), traitCollection.horizontalSizeClass == .regular {
            verticalOffset = 0.0
        }

        let imageInset = UIEdgeInsets(
            top: verticalOffset,
            left: 0.0,
            bottom: -verticalOffset,
            right: 0.0
        )

        for tabBarItem in items ?? [] {
            tabBarItem.title = ""
            tabBarItem.imageInsets = imageInset
        }
    }
}

Solution 8 - Ios

I used the following code in my BaseTabBarController's viewDidLoad. Note that in my example, I have 5 tabs, and selected image will always be base_image + "_selected".

// Get tab bar and set base styles
let tabBar = self.tabBar;
tabBar.backgroundColor = UIColor.whiteColor()
    
// Without this, images can extend off top of tab bar
tabBar.clipsToBounds = true
    
// For each tab item..
let tabBarItems = tabBar.items?.count ?? 0
for i in 0 ..< tabBarItems {
    let tabBarItem = tabBar.items?[i] as UITabBarItem

    // Adjust tab images (Like mstysf says, these values will vary)
    tabBarItem.imageInsets = UIEdgeInsetsMake(5, 0, -6, 0);
    
    // Let's find and set the icon's default and selected states
    // (use your own image names here)
    var imageName = ""
    switch (i) {
    case 0: imageName = "tab_item_feature_1"
    case 1: imageName = "tab_item_feature_2"
    case 2: imageName = "tab_item_feature_3"
    case 3: imageName = "tab_item_feature_4"
    case 4: imageName = "tab_item_feature_5"
    default: break
    }
    tabBarItem.image = UIImage(named:imageName)!.imageWithRenderingMode(.AlwaysOriginal)
    tabBarItem.selectedImage = UIImage(named:imageName + "_selected")!.imageWithRenderingMode(.AlwaysOriginal)
}

Solution 9 - Ios

Swift 4 approach

I was able to do the trick by implementing a function that takes a TabBarItem and does some formatting to it.

Moves the image a little down to make it be more centered and also hides the text of the Tab Bar. Worked better than just setting its title to an empty string, because when you have a NavigationBar as well, the TabBar regains the title of the viewController when selected

func formatTabBarItem(tabBarItem: UITabBarItem){
    tabBarItem.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
    tabBarItem.setTitleTextAttributes([NSAttributedStringKey.foregroundColor:UIColor.clear], for: .selected)
    tabBarItem.setTitleTextAttributes([NSAttributedStringKey.foregroundColor:UIColor.clear], for: .normal)
}

Latest syntax

extension UITabBarItem {
    func setImageOnly(){
        imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
        setTitleTextAttributes([NSAttributedString.Key.foregroundColor:UIColor.clear], for: .selected)
        setTitleTextAttributes([NSAttributedString.Key.foregroundColor:UIColor.clear], for: .normal)
    }
 }

And just use it in your tabBar as:

tabBarItem.setImageOnly()

Solution 10 - Ios

Here is a better, more foolproof way to do this other than the top answer:

[[UITabBarItem appearance] setTitleTextAttributes:@{NSForegroundColorAttributeName: [UIColor clearColor]}
                                         forState:UIControlStateNormal];
[[UITabBarItem appearance] setTitleTextAttributes:@{NSForegroundColorAttributeName: [UIColor clearColor]}
                                         forState:UIControlStateHighlighted];

Put this in your AppDelegate.didFinishLaunchingWithOptions so that it affects all tab bar buttons throughout the life of your app.

Solution 11 - Ios

A minimal, safe UITabBarController extension in Swift (based on @korgx9 answer):

extension UITabBarController {
  func removeTabbarItemsText() {
    tabBar.items?.forEach {
      $0.title = ""
      $0.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
    }
  }
}

Solution 12 - Ios

Based on the answer of ddiego, in Swift 4.2:

extension UITabBarController {
    func cleanTitles() {
        guard let items = self.tabBar.items else {
            return
        }
        for item in items {
            item.title = ""
            item.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
        }
    }
}

And you just need to call self.tabBarController?.cleanTitles() in your view controller.

Solution 13 - Ios

Custom TabBar - iOS 13, Swift 5, XCode 11

  • TabBar items without text
  • TabBar items centered vertically
  • Rounded TabBar view
  • TabBar Dynamic position and frames

Storyboard based. It can be achieved easily programmatically too. Only 4 Steps to follow:

  1. Tab Bar Icons must be in 3 sizes, in black color. Usually, I download from fa2png.io - sizes: 25x25, 50x50, 75x75. PDF image files do not work!

enter image description here

  1. In Storyboard for the tab bar item set the icon you want to use through Attributes Inspector. (see screenshot)

enter image description here

  1. Custom TabBarController -> New File -> Type: UITabBarController -> Set on storyboard. (see screenshot)

enter image description here

  1. UITabBarController class

    class RoundedTabBarViewController: UITabBarController {

    override func viewDidLoad() {
        super.viewDidLoad()
        
        // Do any additional setup after loading the view.
        // Custom tab bar view
        customizeTabBarView()
    }
    
    private func customizeTabBarView() {
        let tabBarHeight = tabBar.frame.size.height
        self.tabBar.layer.masksToBounds = true
        self.tabBar.isTranslucent = true
        self.tabBar.barStyle = .default
        self.tabBar.layer.cornerRadius = tabBarHeight/2
        self.tabBar.layer.maskedCorners = [.layerMaxXMaxYCorner, .layerMaxXMinYCorner, .layerMinXMaxYCorner, .layerMinXMinYCorner]
    }
    
    override func viewDidLayoutSubviews() {
        super.viewDidLayoutSubviews()
        let viewWidth = self.view.bounds.width
        let leadingTrailingSpace = viewWidth * 0.05
        
        tabBar.frame = CGRect(x: leadingTrailingSpace, y: 200, width: viewWidth - (2 * leadingTrailingSpace), height: 49)
    }
    

    }

  2. Result enter image description here

Solution 14 - Ios

enter image description here

code:

private func removeText() {
    if let items = yourTabBarVC?.tabBar.items {
       for item in items {
          item.title = ""
       }
    }
 }

Solution 15 - Ios

Based on all the great answers on this page, I've crafted another solution that also allows you to show the the title again. Instead of removing the content of title, I just change the font color to transparent.

extension UITabBarItem {

	func setTitleColorFor(normalState: UIColor, selectedState: UIColor) {
		self.setTitleTextAttributes([NSAttributedString.Key.foregroundColor: normalState], for: .normal)
		self.setTitleTextAttributes([NSAttributedString.Key.foregroundColor: selectedState], for: .selected)
	}

}


extension UITabBarController {

	func hideItemsTitle() {

		guard let items = self.tabBar.items else {
			return
		}

		for item in items {
			item.setTitleColorFor(normalState: UIColor(white: 0, alpha: 0), selectedState: UIColor(white: 0, alpha: 0))
			item.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
		}

	}

	func showItemsTitle() {

		guard let items = self.tabBar.items else {
			return
		}

		for item in items {
			item.setTitleColorFor(normalState: .black, selectedState: .yellow)
			item.imageInsets = UIEdgeInsets(top: 0, left: 0, bottom: 0, right: 0)
		}

	}

}

Solution 16 - Ios

In my case, same ViewController was used in TabBar and other navigation flow. Inside my ViewController, I have set self.title = "Some Title" which was appearing in TabBar regardless of setting title nil or blank while adding it in tab bar. I have also set imageInsets as follow:

item.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)

So inside my ViewController, I have handled navigation title as follow:

if isFromTabBar {
   // Title for NavigationBar when ViewController is added in TabBar
   // NOTE: Do not set self.title = "Some Title" here as it will set title of tabBarItem
   self.navigationItem.title = "Some Title"
} else {
   // Title for NavigationBar when ViewController is opened from navigation flow
   self.title = "Some Title"
}

Solution 17 - Ios

Easiest way and always works:

class TabBar: UITabBar {
    
    override func layoutSubviews() {
        super.layoutSubviews()
        
        subviews.forEach { subview in
            if subview is UIControl {
                subview.subviews.forEach {
                    if $0 is UILabel {
                        $0.isHidden = true
                        subview.frame.origin.y = $0.frame.height / 2.0
                    }
                }
            }
        }
    }
}

Solution 18 - Ios

make a subclass of UITabBarController and assign that to your tabBar , then in the viewDidLoad method place this line of code:

tabBar.items?.forEach({ (item) in
        item.imageInsets = UIEdgeInsets.init(top: 8, left: 0, bottom: -8, right: 0)
    })

Solution 19 - Ios

If you are looking to center the tabs / change the image insets without using magic numbers, the following has worked for me (in Swift 5.2.2):

In a UITabBarController subclass, you can add add the image insets after setting the view controllers.

override var viewControllers: [UIViewController]? {
    didSet {
      addImageInsets()
    }
}

func addImageInsets() {
    let tabBarHeight = tabBar.frame.height

    for item in tabBar.items ?? [] where item.image != nil {
      let imageHeight = item.image?.size.height ?? 0
      let inset = CGFloat(Int((tabBarHeight - imageHeight) / 4))
      item.imageInsets = UIEdgeInsets(top: inset,
                                      left: 0,
                                      bottom: -inset,
                                      right: 0)
    }
}

Several of the options above list solutions for dealing with hiding the text.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionUpCatView Question on Stackoverflow
Solution 1 - IosmustafaView Answer on Stackoverflow
Solution 2 - IosddiegoView Answer on Stackoverflow
Solution 3 - IosnuynaitView Answer on Stackoverflow
Solution 4 - IosnsinvocationView Answer on Stackoverflow
Solution 5 - IosMarcos ReboucasView Answer on Stackoverflow
Solution 6 - Ioskorgx9View Answer on Stackoverflow
Solution 7 - IosatlwxView Answer on Stackoverflow
Solution 8 - IosFourView Answer on Stackoverflow
Solution 9 - IostapizquentView Answer on Stackoverflow
Solution 10 - IosEzekiel VictorView Answer on Stackoverflow
Solution 11 - IosFederico ZanetelloView Answer on Stackoverflow
Solution 12 - Iosj_gonferView Answer on Stackoverflow
Solution 13 - IosEgzon P.View Answer on Stackoverflow
Solution 14 - IosGiangView Answer on Stackoverflow
Solution 15 - IosladislasView Answer on Stackoverflow
Solution 16 - IosChintan ShahView Answer on Stackoverflow
Solution 17 - IosJulianKroView Answer on Stackoverflow
Solution 18 - Iossoheil pakgoharView Answer on Stackoverflow
Solution 19 - IosCampbell_SoupedView Answer on Stackoverflow