iOS 11 navigation bar height customizing

后端 未结 11 757
无人共我
无人共我 2020-11-28 01:27

Now in iOS 11, the sizeThatFits method is not called from UINavigationBar subclasses. Changing the frame of UINavigationBar causes gli

相关标签:
11条回答
  • 2020-11-28 01:43

    Updated 07 Jan 2018

    This code is support XCode 9.2, iOS 11.2

    I had the same problem. Below is my solution. I assume that height size is 66.

    Please choose my answer if it helps you.

    Create CINavgationBar.swift

       import UIKit
    
    @IBDesignable
    class CINavigationBar: UINavigationBar {
    
        //set NavigationBar's height
        @IBInspectable var customHeight : CGFloat = 66
    
        override func sizeThatFits(_ size: CGSize) -> CGSize {
    
            return CGSize(width: UIScreen.main.bounds.width, height: customHeight)
    
        }
    
        override func layoutSubviews() {
            super.layoutSubviews()
    
            print("It called")
    
            self.tintColor = .black
            self.backgroundColor = .red
    
    
    
            for subview in self.subviews {
                var stringFromClass = NSStringFromClass(subview.classForCoder)
                if stringFromClass.contains("UIBarBackground") {
    
                    subview.frame = CGRect(x: 0, y: 0, width: self.frame.width, height: customHeight)
    
                    subview.backgroundColor = .green
                    subview.sizeToFit()
                }
    
                stringFromClass = NSStringFromClass(subview.classForCoder)
    
                //Can't set height of the UINavigationBarContentView
                if stringFromClass.contains("UINavigationBarContentView") {
    
                    //Set Center Y
                    let centerY = (customHeight - subview.frame.height) / 2.0
                    subview.frame = CGRect(x: 0, y: centerY, width: self.frame.width, height: subview.frame.height)
                    subview.backgroundColor = .yellow
                    subview.sizeToFit()
    
                }
            }
    
    
        }
    
    
    }
    

    Set Storyboard

    Set Custom NavigationBar class

    Add TestView + Set SafeArea

    ViewController.swift

    import UIKit
    
    class ViewController: UIViewController {
    
        var navbar : UINavigationBar!
    
        @IBOutlet weak var testView: UIView!
    
        override func viewDidLoad() {
            super.viewDidLoad()
    
            //update NavigationBar's frame
            self.navigationController?.navigationBar.sizeToFit()
            print("NavigationBar Frame : \(String(describing: self.navigationController!.navigationBar.frame))")
    
        }
    
        //Hide Statusbar
        override var prefersStatusBarHidden: Bool {
    
            return true
        }
    
        override func viewDidAppear(_ animated: Bool) {
    
            super.viewDidAppear(false)
    
            //Important!
            if #available(iOS 11.0, *) {
    
                //Default NavigationBar Height is 44. Custom NavigationBar Height is 66. So We should set additionalSafeAreaInsets to 66-44 = 22
                self.additionalSafeAreaInsets.top = 22
    
            }
    
        }
    
        override func didReceiveMemoryWarning() {
            super.didReceiveMemoryWarning()
            // Dispose of any resources that can be recreated.
        }
    
    
    }
    

    SecondViewController.swift

    import UIKit
    
    class SecondViewController: UIViewController {
    
        override func viewDidLoad() {
            super.viewDidLoad()
    
            // Do any additional setup after loading the view.
    
    
            // Create BackButton
            var backButton: UIBarButtonItem!
            let backImage = imageFromText("Back", font: UIFont.systemFont(ofSize: 16), maxWidth: 1000, color:UIColor.white)
            backButton = UIBarButtonItem(image: backImage, style: UIBarButtonItemStyle.plain, target: self, action: #selector(SecondViewController.back(_:)))
    
            self.navigationItem.leftBarButtonItem = backButton
            self.navigationItem.leftBarButtonItem?.setBackgroundVerticalPositionAdjustment(-10, for: UIBarMetrics.default)
    
    
        }
        override var prefersStatusBarHidden: Bool {
    
            return true
        }
        @objc func back(_ sender: UITabBarItem){
    
            self.navigationController?.popViewController(animated: true)
    
        }
    
    
        //Helper Function : Get String CGSize
        func sizeOfAttributeString(_ str: NSAttributedString, maxWidth: CGFloat) -> CGSize {
            let size = str.boundingRect(with: CGSize(width: maxWidth, height: 1000), options:(NSStringDrawingOptions.usesLineFragmentOrigin), context:nil).size
            return size
        }
    
    
        //Helper Function : Convert String to UIImage
        func imageFromText(_ text:NSString, font:UIFont, maxWidth:CGFloat, color:UIColor) -> UIImage
        {
            let paragraph = NSMutableParagraphStyle()
            paragraph.lineBreakMode = NSLineBreakMode.byWordWrapping
            paragraph.alignment = .center // potentially this can be an input param too, but i guess in most use cases we want center align
    
            let attributedString = NSAttributedString(string: text as String, attributes: [NSAttributedStringKey.font: font, NSAttributedStringKey.foregroundColor: color, NSAttributedStringKey.paragraphStyle:paragraph])
    
            let size = sizeOfAttributeString(attributedString, maxWidth: maxWidth)
            UIGraphicsBeginImageContextWithOptions(size, false , 0.0)
            attributedString.draw(in: CGRect(x: 0, y: 0, width: size.width, height: size.height))
            let image = UIGraphicsGetImageFromCurrentImageContext()
            UIGraphicsEndImageContext()
            return image!
        }
    
    
    
    
        override func didReceiveMemoryWarning() {
            super.didReceiveMemoryWarning()
            // Dispose of any resources that can be recreated.
        }
    
    
    
    }
    

    Yellow is barbackgroundView. Black opacity is BarContentView.

    And I removed BarContentView's backgroundColor.

    That's It.

    0 讨论(0)
  • 2020-11-28 01:45

    this works for me :

    - (CGSize)sizeThatFits:(CGSize)size {
        CGSize sizeThatFit = [super sizeThatFits:size];
        if ([UIApplication sharedApplication].isStatusBarHidden) {
            if (sizeThatFit.height < 64.f) {
                sizeThatFit.height = 64.f;
            }
        }
        return sizeThatFit;
    }
    
    - (void)setFrame:(CGRect)frame {
        if ([UIApplication sharedApplication].isStatusBarHidden) {
            frame.size.height = 64;
        }
        [super setFrame:frame];
    }
    
    - (void)layoutSubviews
    {
        [super layoutSubviews];
    
        for (UIView *subview in self.subviews) {
            if ([NSStringFromClass([subview class]) containsString:@"BarBackground"]) {
                CGRect subViewFrame = subview.frame;
                subViewFrame.origin.y = 0;
                subViewFrame.size.height = 64;
                [subview setFrame: subViewFrame];
            }
            if ([NSStringFromClass([subview class]) containsString:@"BarContentView"]) {
                CGRect subViewFrame = subview.frame;
                subViewFrame.origin.y = 20;
                subViewFrame.size.height = 44;
                [subview setFrame: subViewFrame];
            }
        }
    }
    
    0 讨论(0)
  • This works well for the regular navigation bar. If your using the LargeTitle this wont work well because the titleView size isn't going to be a fixed height of 44 points. But for the regular view this should be suffice.

    Like @frangulyan apple suggested to add a view beneath the navBar and hide the thin line (shadow image). This is what I came up with below. I added an uiview to the navigationItem's titleView and then added an imageView inside that uiview. I removed the thin line (shadow image). The uiview I added is the same exact color as the navBar. I added a uiLabel inside that view and that's it.

    Here's the 3d image. The extended view is behind the usernameLabel underneath the navBar. Its gray and has a thin line underneath of it. Just anchor your collectionView or whatever underneath of the thin separatorLine.

    The 9 steps are explained above each line of code:

    class ExtendedNavController: UIViewController {
    
        fileprivate let extendedView: UIView = {
            let view = UIView()
            view.translatesAutoresizingMaskIntoConstraints = false
            view.backgroundColor = .white
            return view
        }()
    
        fileprivate let separatorLine: UIView = {
            let view = UIView()
            view.translatesAutoresizingMaskIntoConstraints = false
            view.backgroundColor = .gray
            return view
        }()
    
        fileprivate let usernameLabel: UILabel = {
            let label = UILabel()
            label.translatesAutoresizingMaskIntoConstraints = false
            label.font = UIFont.systemFont(ofSize: 14)
            label.text = "username goes here"
            label.textAlignment = .center
            label.lineBreakMode = .byTruncatingTail
            label.numberOfLines = 1
            return label
        }()
    
        fileprivate let myTitleView: UIView = {
            let view = UIView()
            view.backgroundColor = .white
            return view
        }()
    
        fileprivate let profileImageView: UIImageView = {
            let imageView = UIImageView()
            imageView.translatesAutoresizingMaskIntoConstraints = false
            imageView.clipsToBounds = true
            imageView.backgroundColor = .darkGray
            return imageView
        }()
    
        override func viewDidLoad() {
            super.viewDidLoad()
            view.backgroundColor = .white
    
            // 1. the navBar's titleView has a height of 44, set myTitleView height and width both to 44
            myTitleView.frame = CGRect(x: 0, y: 0, width: 44, height: 44)
    
            // 2. set myTitleView to the nav bar's titleView
            navigationItem.titleView = myTitleView
    
            // 3. get rid of the thin line (shadow Image) underneath the navigationBar
            navigationController?.navigationBar.setValue(true, forKey: "hidesShadow")
            navigationController?.navigationBar.layoutIfNeeded()
    
            // 4. set the navigationBar's tint color to the color you want
            navigationController?.navigationBar.barTintColor = UIColor(red: 249.0/255.0, green: 249.0/255.0, blue: 249.0/255.0, alpha: 1.0)
    
            // 5. set extendedView's background color to the same exact color as the navBar's background color
            extendedView.backgroundColor = UIColor(red: 249.0/255.0, green: 249.0/255.0, blue: 249.0/255.0, alpha: 1.0)
    
            // 6. set your imageView to get pinned inside the titleView
            setProfileImageViewAnchorsInsideMyTitleView()
    
            // 7. set the extendedView's anchors directly underneath the navigation bar
            setExtendedViewAndSeparatorLineAnchors()
    
            // 8. set the usernameLabel's anchors inside the extendedView
            setNameLabelAnchorsInsideTheExtendedView()
        }
    
        override func viewWillDisappear(_ animated: Bool) {
            super.viewWillDisappear(true)
    
            // 9. **Optional** If you want the shadow image to show on other view controllers when popping or pushing
            navigationController?.navigationBar.setBackgroundImage(nil, for: .default)
            navigationController?.navigationBar.setValue(false, forKey: "hidesShadow")
            navigationController?.navigationBar.layoutIfNeeded()
        }
    
        func setExtendedViewAndSeparatorLineAnchors() {
    
            view.addSubview(extendedView)
            view.addSubview(separatorLine)
    
            extendedView.topAnchor.constraint(equalTo: view.safeAreaLayoutGuide.topAnchor).isActive = true
            extendedView.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
            extendedView.trailingAnchor.constraint(equalTo: view.trailingAnchor).isActive = true
            extendedView.heightAnchor.constraint(equalToConstant: 29.5).isActive = true
    
            separatorLine.topAnchor.constraint(equalTo:  extendedView.bottomAnchor).isActive = true
            separatorLine.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
            separatorLine.trailingAnchor.constraint(equalTo: view.trailingAnchor).isActive = true
            separatorLine.heightAnchor.constraint(equalToConstant: 0.5).isActive = true
        }
    
        func setProfileImageViewAnchorsInsideMyTitleView() {
    
            myTitleView.addSubview(profileImageView)
    
            profileImageView.topAnchor.constraint(equalTo: myTitleView.topAnchor).isActive = true
            profileImageView.centerXAnchor.constraint(equalTo: myTitleView.centerXAnchor).isActive = true
            profileImageView.widthAnchor.constraint(equalToConstant: 44).isActive = true
            profileImageView.heightAnchor.constraint(equalToConstant: 44).isActive = true
    
            // round the profileImageView
            profileImageView.layoutIfNeeded()
            profileImageView.layer.cornerRadius = profileImageView.frame.width / 2
        }
    
        func setNameLabelAnchorsInsideTheExtendedView() {
    
            extendedView.addSubview(usernameLabel)
    
            usernameLabel.topAnchor.constraint(equalTo: extendedView.topAnchor).isActive = true
            usernameLabel.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
            usernameLabel.trailingAnchor.constraint(equalTo: view.trailingAnchor).isActive = true
        }
    }
    
    0 讨论(0)
  • 2020-11-28 01:49

    I was doubling the height of my navigation bar so I could add a row of status icons above the default navigation controls, by subclassing UINavigationBar and using sizeThatFits to override the height. Fortunately this has the same effect, and is simpler, with fewer side effects. I tested it with iOS 8 through 11. Put this in your view controller:

    - (void)viewDidLoad {
        [super viewDidLoad];
        if (self.navigationController) {
            self.navigationItem.prompt = @" "; // this adds empty space on top
        }
    }
    
    0 讨论(0)
  • 2020-11-28 01:51

    According to Apple developers (look here, here and here), changing navigation bar height in iOS 11 is not supported. Here they suggest to do workarounds like having a view under the navigation bar (but outside of it) and then remove the nav bar border. As a result, you will have this in storyboard:

    look like this on the device:

    Now you can do a workaround that was suggested in the other answers: create a custom subclass of UINavigationBar, add your custom large subview to it, override sizeThatFits and layoutSubviews, then set additionalSafeAreaInsets.top for the navigation's top controller to the difference customHeight - 44px, but the bar view will still be the default 44px, even though visually everything will look perfect. I didn't try overriding setFrame, maybe it works, however, as Apple developer wrote in one of the links above: "...and neither is [supported] changing the frame of a navigation bar that is owned by a UINavigationController (the navigation controller will happily stomp on your frame changes whenever it deems fit to do so)."

    In my case the above workaround made views to look like this (debug view to show borders):

    As you can see, the visual appearance is quite good, the additionalSafeAreaInsets correctly pushed the content down, the big navigation bar is visible, however I have a custom button in this bar and only the area that goes under the standard 44 pixel nav bar is clickable (green area in the image). Touches below the standard navigation bar height doesn't reach my custom subview, so I need the navigation bar itself to be resized, which the Apple developers say is not supported.

    0 讨论(0)
  • 2020-11-28 01:51

    Along with overriding -layoutSubviews and -setFrame: you should check out the newly added UIViewController's additionalSafereaInsets property (Apple Documentation) if you do not want the resized navigation bar hiding your content.

    0 讨论(0)
提交回复
热议问题