在界面构建器中,按住 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];
答案 0 :(得分:126)
在UIKit中,NSString类增加了从给定的NSString对象获取当以某种字体呈现时它将占用的大小。
简而言之,如果你去:
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完成此操作,请尝试设置可变宽度约束:
您可能还需要调整Content Hugging Priority
和Content Compression Resistance Priority
以获得所需的结果。
UILabel 完全自动调整大小:
这个UILabel只是设置在屏幕中心(仅限两个约束,水平/垂直):
它会自动更改宽度:
您无需设置任何宽度或高度 - 这是完全自动的。
注意简单地附上小黄色方块(“间距”为零)。它们会在UILabel调整大小时自动移动。
添加“&gt; =”约束设置UILabel的最小宽度:
答案 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及更高版本中,现在可以使用“自动布局/约束”来完成此操作。
主要优点是:
一些缺点:
最酷的是我们专注于声明一个意图,例如:
Here是一个介绍自动布局的简单教程。
对于more details。
一开始需要一些时间,但看起来确实值得努力。
答案 7 :(得分:6)
sizeToFit无法正常工作。代替:
myButton.size = myButton.sizeThatFits(CGSize.zero)
您还可以在按钮上添加contentInset
:
myButton.contentEdgeInsets = UIEdgeInsetsMake(8, 8, 4, 8)
答案 8 :(得分:5)
简单地:
UIView
作为自动布局包装的包装器。 UILabel
放入该包装器中。添加将tyour标签粘贴到包装器边缘的约束。UIButton
放入包装器中,然后简单地添加与UILabel
相同的约束。答案 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
内的按钮,并使用自动布局。
对我有用的是:
intrinsicContentSize
的宽度,因为根据此document自动布局,尚不知道intrinsicContentSize
。将宽度约束设置为intrinsicContentSize
宽度答案 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)
说实话,我认为在情节提要中没有简单的复选框来表明您想要调整按钮的大小以容纳文本真是可耻。好吧...
这是使用情节提要的最简单解决方案。
将UILabel放置在UIView中。设置约束以将其附加到UIView的边缘。
将您的UIButton放在UIView中。设置相同的约束以将其附加到UIView的边缘。
设置插座。
@IBOutlet var button: UIButton!
@IBOutlet var textOnTheButton: UILabel!
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."
override func viewDidLoad() {
super.viewDidLoad()
textOnTheButton.text = someTitle
button.setTitle(someTitle, for: .normal)
button.titleLabel?.numberOfLines = 0
}