iOS:UIButton根据文本长度调整大小

时间:2010-11-09 14:53:05

标签: ios uibutton

在界面构建器中,按住 Command + = 将调整按钮的大小以适合其文本。我想知道在按钮添加到视图之前是否可以以编程方式执行此操作。

UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
[button.titleLabel setFont:[UIFont fontWithName:@"Arial-BoldMT" size:12]];
[button addTarget:self action:@selector(buttonTapped:) forControlEvents:UIControlEventTouchUpInside];
// I need to know the width needed to accomodate NSStringVariable
[button setTitle:NSStringVariable forState:UIControlStateNormal]; 
// So that I can set the width property of the button before addSubview
[button setFrame:CGRectMake(10, 0, width, fixedHeight)];
[subNavigation addSubview:button];

14 个答案:

答案 0 :(得分:126)

在UIKit中,NSString类增加了从给定的NSString对象获取当以某种字体呈现时它将占用的大小。

文档是here。现在它已被弃用here

简而言之,如果你去:

CGSize stringsize = [myString sizeWithFont:[UIFont systemFontOfSize:14]]; 
//or whatever font you're using
[button setFrame:CGRectMake(10,0,stringsize.width, stringsize.height)];

...您将按钮的框架设置为您正在渲染的字符串的高度和宽度。

你可能想在CGSize周围尝试一些缓冲空间,但你会在正确的位置开始。

答案 1 :(得分:94)

在代码中执行此操作的方法是:

[button sizeToFit];

如果您是子类并希望添加额外规则,则可以覆盖:

- (CGSize)sizeThatFits:(CGSize)size;

答案 2 :(得分:31)

如果您的按钮是使用Interface Builder制作的,并且您要在代码中更改标题,则可以执行以下操作:

[self.button setTitle:@"Button Title" forState:UIControlStateNormal];
[self.button sizeToFit];

答案 3 :(得分:19)

<强>夫特:

重要的是你会调用apply_to_zero_and_increment g时,它应该是在设置要显示的文本之后它可以读取所需的大小,如果你以后更新文本,那么再打一次。

.sizeToFit()

答案 4 :(得分:14)

要使用autolayout完成此操作,请尝试设置可变宽度约束:

Width Constraint

您可能还需要调整Content Hugging PriorityContent Compression Resistance Priority以获得所需的结果。

UILabel 完全自动调整大小

这个UILabel只是设置在屏幕中心(仅限两个约束,水平/垂直):

它会自动更改宽度:

您无需设置任何宽度或高度 - 这是完全自动的。

enter image description here

enter image description here

注意简单地附上小黄色方块(“间距”为零)。它们会在UILabel调整大小时自动移动。

添加“&gt; =”约束设置UILabel的最小宽度:

enter image description here

答案 5 :(得分:13)

如果要调整文本大小而不是按钮,可以使用...

button.titleLabel.adjustsFontSizeToFitWidth = YES;
button.titleLabel.minimumScaleFactor = .5;
// The .5 value means that I will let it go down to half the original font size 
// before the texts gets truncated

// note, if using anything pre ios6, you would use I think
button.titleLabel.minimumFontSize = 8;

答案 6 :(得分:7)

在Xcode 4.5及更高版本中,现在可以使用“自动布局/约束”来完成此操作。

主要优点是:

  1. 您根本不需要以编程方式设置框架!
  2. 如果操作正确,您无需为了更改方向而重置帧。
  3. 此外,设备更改无需打扰您(阅读,无需针对不同的屏幕尺寸单独编码)。
  4. 一些缺点:

    1. 不向后兼容 - 仅适用于iOS 6及更高版本。
    2. 需要熟悉(但以后会节省时间)。
    3. 最酷的是我们专注于声明一个意图,例如:

      • 我希望这两个按钮具有相同的宽度;或
      • 我需要这个视图垂直居中,并从superview的边缘延伸到10 pts的最大值;甚至,
      • 我希望此按钮/标签根据显示的标签调整大小!

      Here是一个介绍自动布局的简单教程。

      对于more details

      一开始需要一些时间,但看起来确实值得努力。

答案 7 :(得分:6)

sizeToFit无法正常工作。代替:

myButton.size = myButton.sizeThatFits(CGSize.zero)

您还可以在按钮上添加contentInset

myButton.contentEdgeInsets = UIEdgeInsetsMake(8, 8, 4, 8)

答案 8 :(得分:5)

简单地:

  1. 创建UIView作为自动布局包装的包装器。
  2. UILabel放入该包装器中。添加将tyour标签粘贴到包装器边缘的约束。
  3. UIButton放入包装器中,然后简单地添加与UILabel相同的约束。
  4. 享受自动调整按钮和文字。

答案 9 :(得分:2)

我还有一些与sizeToFit无法解决的帖子相关的其他需求。无论文字如何,我都需要将按钮置于其原始位置的中心位置。我也从不希望按钮大于原始尺寸。

因此,我将按钮布局为其最大尺寸,在Controller中保存该尺寸,并在文本更改时使用以下方法调整按钮的大小:

+ (void) sizeButtonToText:(UIButton *)button availableSize:(CGSize)availableSize padding:(UIEdgeInsets)padding {    

     CGRect boundsForText = button.frame;

    // Measures string
    CGSize stringSize = [button.titleLabel.text sizeWithFont:button.titleLabel.font];
    stringSize.width = MIN(stringSize.width + padding.left + padding.right, availableSize.width);

    // Center's location of button
    boundsForText.origin.x += (boundsForText.size.width - stringSize.width) / 2;
    boundsForText.size.width = stringSize.width;
    [button setFrame:boundsForText];
 }

答案 10 :(得分:2)

由于某些原因,func sizeToFit()对我不起作用。我的设置是使用UITableViewCell内的按钮,并使用自动布局。

对我有用的是:

  1. 获取宽度约束
  2. 获得intrinsicContentSize的宽度,因为根据此document自动布局,尚不知道intrinsicContentSize。将宽度约束设置为intrinsicContentSize宽度

enter image description here

这是Debug View Hierachry的两个标题 enter image description here

enter image description here

答案 11 :(得分:1)

此按钮类具有自动调整高度的文本(对于Xamarin,但可以用其他语言重写)

[Register(nameof(ResizableButton))]
public class ResizableButton : UIButton
{
    NSLayoutConstraint _heightConstraint;
    public bool NeedsUpdateHeightConstraint { get; private set; } = true;

    public ResizableButton(){}

    public ResizableButton(UIButtonType type) : base(type){}

    public ResizableButton(NSCoder coder) : base(coder){}

    public ResizableButton(CGRect frame) : base(frame){}

    protected ResizableButton(NSObjectFlag t) : base(t){}

    protected internal ResizableButton(IntPtr handle) : base(handle){}

    public override void LayoutSubviews()
    {
        base.LayoutSubviews();
        UpdateHeightConstraint();
        InvalidateIntrinsicContentSize();
    }

    public override void SetTitle(string title, UIControlState forState)
    {
        NeedsUpdateHeightConstraint = true;
        base.SetTitle(title, forState);
    }

    private void UpdateHeightConstraint()
    {
        if (!NeedsUpdateHeightConstraint)
            return;
        NeedsUpdateHeightConstraint = false;
        var labelSize = TitleLabel.SizeThatFits(new CGSize(Frame.Width - TitleEdgeInsets.Left - TitleEdgeInsets.Right, float.MaxValue));

        var rect = new CGRect(Frame.X, Frame.Y, Frame.Width, labelSize.Height + TitleEdgeInsets.Top + TitleEdgeInsets.Bottom);

        if (_heightConstraint != null)
            RemoveConstraint(_heightConstraint);

        _heightConstraint = NSLayoutConstraint.Create(this, NSLayoutAttribute.Height, NSLayoutRelation.Equal, 1, rect.Height);
        AddConstraint(_heightConstraint);
    }

     public override CGSize IntrinsicContentSize
     {
         get
         {
             var labelSize = TitleLabel.SizeThatFits(new CGSize(Frame.Width - TitleEdgeInsets.Left - TitleEdgeInsets.Right, float.MaxValue));
             return new CGSize(Frame.Width, labelSize.Height + TitleEdgeInsets.Top + TitleEdgeInsets.Bottom);
         }
     }
}

答案 12 :(得分:0)

Swift 4.2

谢谢上帝,这解决了。在按钮上设置文本后,您可以从UIView检索固有的ContentSize,它是自然大小(正式文档为here)。 对于UIButton,您可以按以下方式使用它。

button.intrinsicContentSize.width

为您提供信息,我调整了宽度以使其看起来正确。

button.frame = CGRect(fx: xOffset, y: 0.0, width: button.intrinsicContentSize.width + 18, height: 40)

模拟器

UIButtons with intrinsicContentSize

来源:https://riptutorial.com/ios/example/16418/get-uibutton-s-size-strictly-based-on-its-text-and-font

答案 13 :(得分:0)

说实话,我认为在情节提要中没有简单的复选框来表明您想要调整按钮的大小以容纳文本真是可耻。好吧...

这是使用情节提要的最简单解决方案。

  1. 放置UIView并为其添加约束。例: enter image description here
  2. 将UILabel放置在UIView中。设置约束以将其附加到UIView的边缘。

  3. 将您的UIButton放在UIView中。设置相同的约束以将其附加到UIView的边缘。

  4. 将UILabel的行数设置为0。 enter image description here

  5. 设置插座。

    @IBOutlet var button: UIButton!
    @IBOutlet var textOnTheButton: UILabel!
  1. 获得一些长而长的标题。

    let someTitle = "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum."
  1. 在viewDidLoad上,设置UILabel和UIButton的标题。

    override func viewDidLoad() {
        super.viewDidLoad()
        textOnTheButton.text = someTitle
        button.setTitle(someTitle, for: .normal)
        button.titleLabel?.numberOfLines = 0
    }
  1. 运行它以确保调整按钮的大小并可以按下。

enter image description here