いくつかの文字を表示するUILabelがあります。「x」、「y」、「rpm」のように。ラベルのテキストの幅を計算するにはどうすればよいですか(使用可能なスペース全体を使用するわけではありません)。これは自動レイアウト用であり、UILabelの内部に小さいテキストがある場合、別のビューのフレームの長方形が大きくなります。UIFontとフォントサイズが指定されているときにテキストの幅を計算する方法はありますか?また、改行はなく、1行だけです。
いくつかの文字を表示するUILabelがあります。「x」、「y」、「rpm」のように。ラベルのテキストの幅を計算するにはどうすればよいですか(使用可能なスペース全体を使用するわけではありません)。これは自動レイアウト用であり、UILabelの内部に小さいテキストがある場合、別のビューのフレームの長方形が大きくなります。UIFontとフォントサイズが指定されているときにテキストの幅を計算する方法はありますか?また、改行はなく、1行だけです。
回答:
NSString UIKitAdditionsのさまざまなsizeWithFont:メソッドを介して正確にそれを行うことができます。あなたの場合、最も単純なバリアントで十分です(複数行のラベルがないため):
NSString *someString = @"Hello World";
UIFont *yourFont = // [UIFont ...]
CGSize stringBoundingBox = [someString sizeWithFont:yourFont];
この方法にはいくつかのバリエーションがあります。改行モードまたは最大サイズを検討する人もいます。
sizeWithFontは非推奨になっているので、Swift4とを使用するための元の回答を更新します。.size
//: Playground - noun: a place where people can play
import UIKit
if let font = UIFont(name: "Helvetica", size: 24) {
   let fontAttributes = [NSAttributedStringKey.font: font]
   let myText = "Your Text Here"
   let size = (myText as NSString).size(withAttributes: fontAttributes)
}
サイズは、「YourTextHere」の画面上のサイズ(ポイント単位)である必要があります。
sizeWithFont:現在は非推奨です。sizeWithAttributes:代わりに使用してください。
UIFont *font = [UIFont fontWithName:@"Helvetica" size:30];
NSDictionary *userAttributes = @{NSFontAttributeName: font,
                                 NSForegroundColorAttributeName: [UIColor blackColor]};
NSString *text = @"hello";
...
const CGSize textSize = [text sizeWithAttributes: userAttributes];
sizeWithAttributes:メソッドは小数サイズを返します。返されたサイズを使用してビューのサイズを変更するには、ceil関数を使用してその値を最も近い整数に上げる必要があります。
                    この答えは、新しい構文を使用してそれを行うためのはるかにクリーンな方法です。
基づいて、グレン・ハウズ優秀な答えは、私は、文字列の幅を計算するために拡張機能を作成しました。の幅を設定するようなことをしている場合UISegmentedControl、これはセグメントのタイトル文字列に基づいて幅を設定できます。
extension String {
    func widthOfString(usingFont font: UIFont) -> CGFloat {
        let fontAttributes = [NSAttributedString.Key.font: font]
        let size = self.size(withAttributes: fontAttributes)
        return size.width
    }
    func heightOfString(usingFont font: UIFont) -> CGFloat {
        let fontAttributes = [NSAttributedString.Key.font: font]
        let size = self.size(withAttributes: fontAttributes)
        return size.height
    }
    func sizeOfString(usingFont font: UIFont) -> CGSize {
        let fontAttributes = [NSAttributedString.Key.font: font]
        return self.size(withAttributes: fontAttributes)
    }
}
使用法:
    // Set width of segmentedControl
    let starString = "⭐️"
    let starWidth = starString.widthOfString(usingFont: UIFont.systemFont(ofSize: 14)) + 16
    segmentedController.setWidth(starWidth, forSegmentAt: 3)
Swiftのこの単純な拡張機能はうまく機能します。
extension String {
    func size(OfFont font: UIFont) -> CGSize {
        return (self as NSString).size(attributes: [NSFontAttributeName: font])
    }
}
使用法:
let string = "hello world!"
let font = UIFont.systemFont(ofSize: 12)
let width = string.size(OfFont: font).width // size: {w: 98.912 h: 14.32}
スイフト2.3バージョン用です。文字列の幅を取得できます。
var sizeOfString = CGSize()
if let font = UIFont(name: "Helvetica", size: 14.0)
    {
        let finalDate = "Your Text Here"
        let fontAttributes = [NSFontAttributeName: font] // it says name, but a UIFont works
        sizeOfString = (finalDate as NSString).sizeWithAttributes(fontAttributes)
    }
あなたは、テキストを取得するために苦労している場合は幅をして、複数行のサポートあなたは、次のコード(使用できるように、スウィフト5):
func width(text: String, height: CGFloat) -> CGFloat {
    let attributes: [NSAttributedString.Key: Any] = [
        .font: UIFont.systemFont(ofSize: 17)
    ]
    let attributedText = NSAttributedString(string: text, attributes: attributes)
    let constraintBox = CGSize(width: .greatestFiniteMagnitude, height: height)
    let textWidth = attributedText.boundingRect(with: constraintBox, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil).width.rounded(.up)
    return textWidth
}
そして、必要に応じてテキストの高さを見つけることができるのと同じ方法(constraintBoxの実装を切り替えるだけ):
let constraintBox = CGSize(width: maxWidth, height: .greatestFiniteMagnitude)
または、複数行をサポートしてテキストサイズを取得するための統合関数は次のとおりです。
func labelSize(for text: String, maxWidth: CGFloat, maxHeight: CGFloat) -> CGSize {
    let attributes: [NSAttributedString.Key: Any] = [
        .font: UIFont.systemFont(ofSize: 17)
    ]
    let attributedText = NSAttributedString(string: text, attributes: attributes)
    let constraintBox = CGSize(width: maxWidth, height: maxHeight)
    let rect = attributedText.boundingRect(with: constraintBox, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil).integral
    return rect.size
}
使用法:
let textSize = labelSize(for: "SomeText", maxWidth: contentView.bounds.width, maxHeight: .greatestFiniteMagnitude)
let textHeight = textSize.height.rounded(.up)
let textWidth = textSize.width.rounded(.up)
用スウィフト3.0+
extension String {
    func SizeOf_String( font: UIFont) -> CGSize {
        let fontAttribute = [NSFontAttributeName: font]
        let size = self.size(attributes: fontAttribute)  // for Single Line
       return size;
   }
}
次のように使用してください...
        let Str = "ABCDEF"
        let Font =  UIFont.systemFontOfSize(19.0)
        let SizeOfString = Str.SizeOfString(font: Font!)
これがどれほど効率的かはわかりませんが、指定された幅に文字列を収めるポイントサイズを返すこの関数を作成しました。
func fontSizeThatFits(targetWidth: CGFloat, maxFontSize: CGFloat, font: UIFont) -> CGFloat {
    var variableFont = font.withSize(maxFontSize)
    var currentWidth = self.size(withAttributes: [NSAttributedString.Key.font:variableFont]).width
    while currentWidth > targetWidth {
        variableFont = variableFont.withSize(variableFont.pointSize - 1)
        currentWidth = self.size(withAttributes: [NSAttributedString.Key.font:variableFont]).width
    }
    return variableFont.pointSize
}
そしてそれはこのように使用されます:
textView.font = textView.font!.withSize(textView.text!.fontSizeThatFits(targetWidth: view.frame.width, maxFontSize: 50, font: textView.font!))
私のコードのやり方は、UIFontの拡張を作成することです:(これはSwift 4.1です)
extension UIFont {
    public func textWidth(s: String) -> CGFloat
    {
        return s.size(withAttributes: [NSAttributedString.Key.font: self]).width
    }
} // extension UIFont