如何以固定宽度获取NSAttributedString的高度

问题描述 投票:48回答:12

我想在固定宽度的框中绘制一些NSAttributedString,但是在计算绘制它们时会占用的正确高度时遇到问题。到目前为止,我已经尝试过:

  1. 调用- (NSSize) size,但结果没用(为此目的),因为他们会给出字符串所需的宽度。
  2. 用我想要的宽度调整- (void)drawWithRect:(NSRect)rect options:(NSStringDrawingOptions)options和选项中的NSStringDrawingUsesLineFragmentOrigin,就像我在绘图中使用的那样。结果......难以理解;当然不是我想要的。 (正如许多地方所指出的那样,包括this Cocoa-Dev线程)。
  3. 创建临时NSTextView并执行: [[tmpView textStorage] setAttributedString:aString];
    [tmpView setHorizontallyResizable:NO];
    [tmpView sizeToFit];
    当我查询tmpView的框架时,宽度仍然是所希望的,并且高度通常是正确的...直到我得到更长的字符串,当它通常是所需尺寸的一半时。 (似乎没有最大尺寸被击中:一帧将是273.0高(约300太短),另一帧将是478.0(只有60-ish太短))。

如果有人管理过这个,我会感激任何指点。

cocoa nstextview appkit nsattributedstring
12个回答
31
投票
-[NSAttributedString boundingRectWithSize:options:]

你可以指定NSStringDrawingUsesDeviceMetrics来获得union of all glyph bounds

-[NSAttributedString size]不同,返回的NSRect表示如果绘制字符串将会改变的区域的尺寸。

正如@Bryan评论,在OS X 10.11及更高版本中不推荐使用boundingRectWithSize:options:(不推荐)。这是因为字符串样式现在是动态的,具体取决于上下文。

对于OS X 10.11及更高版本,请参阅Apple的Calculating Text Height开发人员文档。


0
投票

正如上面提到的很多人,并根据我的测试。

我在iOS上使用open func boundingRect(with size: CGSize, options: NSStringDrawingOptions = [], context: NSStringDrawingContext?) -> CGRect,如下所示:

let rect = attributedTitle.boundingRect(with: CGSize(width:200, height:0), options: NSStringDrawingOptions.usesLineFragmentOrigin, context: nil)

这里200是固定宽度,因为你的预期,高度我给它0,因为我认为最好是善意告诉API高度是无限的。

选项在这里不是那么重要,我尝试了.usesLineFragmentOrigin.usesLineFragmentOrigin.union(.usesFontLeading).usesLineFragmentOrigin.union(.usesFontLeading).union(.usesDeviceMetrics),它给出了相同的结果。

结果是我的预期。

谢谢。


0
投票

我找到帮助类来查找attributedText的高度和宽度(测试代码)

https://gist.github.com/azimin/aa1a79aefa1cec031152fa63401d2292

在项目中添加以上文件

如何使用

let attribString = AZTextFrameAttributes(attributedString: lbl.attributedText!)
let width : CGFloat = attribString.calculatedTextWidth()
print("width is :: >> \(width)")

0
投票

Swift 4.2

let attributedString = self.textView.attributedText
let rect = attributedString?.boundingRect(with: CGSize(width: self.textView.frame.width, height: CGFloat.greatestFiniteMagnitude), options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil)
print("attributedString Height = ",rect?.height)

15
投票

答案是使用 - (void)drawWithRect:(NSRect)rect options:(NSStringDrawingOptions)options 但是你传入的rect应该在你想要无限的维度中有0.0(呃,这是完全合理的)。例子here


6
投票

您可能对Jerry Krinock的伟大(仅限OS X)NS(Attributed)String+Geometrics类别感兴趣,该类别旨在进行各种字符串测量,包括您正在寻找的内容。


6
投票

我有一个复杂的属性字符串与多个字体,并得到不正确的结果与我尝试的一些上述答案。使用UITextView给了我正确的高度,但对于我的用例来说太慢了(调整收集单元格)。我使用之前引用并由Erik描述的Apple doc中描述的相同的一般方法编写了快速代码。这给了我正确的结果,必须比UITextView进行计算更快的执行。

private func heightForString(_ str : NSAttributedString, width : CGFloat) -> CGFloat {
    let ts = NSTextStorage(attributedString: str)

    let size = CGSize(width:width, height:CGFloat.greatestFiniteMagnitude)

    let tc = NSTextContainer(size: size)
    tc.lineFragmentPadding = 0.0

    let lm = NSLayoutManager()
    lm.addTextContainer(tc)

    ts.addLayoutManager(lm)
    lm.glyphRange(forBoundingRect: CGRect(origin: .zero, size: size), in: tc)

    let rect = lm.usedRect(for: tc)

    return rect.integral.size.height
}

2
投票

使用NSAttributedString方法

- (CGRect)boundingRectWithSize:(CGSize)size options:(NSStringDrawingOptions)options context:(NSStringDrawingContext *)context

大小是区域的约束,计算的区域宽度限制为指定的宽度,而高度基于此宽度是灵活的。如果不可用,可以为上下文指定nil。要获得多行文本大小,请使用NSStringDrawingUsesLineFragmentOrigin作为选项。


2
投票

在OS X 10.11+上,以下方法适用于我(来自Apple的Calculating Text Height文档)

- (CGFloat)heightForString:(NSAttributedString *)myString atWidth:(float)myWidth
{
    NSTextStorage *textStorage = [[NSTextStorage alloc] initWithAttributedString:myString];
    NSTextContainer *textContainer = [[NSTextContainer alloc] initWithContainerSize:
        NSMakeSize(myWidth, FLT_MAX)];
    NSLayoutManager *layoutManager = [[NSLayoutManager alloc] init];
    [layoutManager addTextContainer:textContainer];
    [textStorage addLayoutManager:layoutManager];
    [layoutManager glyphRangeForTextContainer:textContainer];
    return [layoutManager
        usedRectForTextContainer:textContainer].size.height;
}

2
投票

斯威夫特3:

let attributedStringToMeasure = NSAttributedString(string: textView.text, attributes: [
        NSFontAttributeName: UIFont(name: "GothamPro-Light", size: 15)!,
        NSForegroundColorAttributeName: ClickUpConstants.defaultBlackColor
])

let placeholderTextView = UITextView(frame: CGRect(x: 0, y: 0, width: widthOfActualTextView, height: 10))
placeholderTextView.attributedText = attributedStringToMeasure
let size: CGSize = placeholderTextView.sizeThatFits(CGSize(width: widthOfActualTextView, height: CGFloat.greatestFiniteMagnitude))

height = size.height

这个答案对我来说很有用,不像其他那些给我更大的字符串不正确的高度。

如果要使用常规文本而不是归因文本执行此操作,请执行以下操作:

let placeholderTextView = UITextView(frame: CGRect(x: 0, y: 0, width: ClickUpConstants.screenWidth - 30.0, height: 10))
placeholderTextView.text = "Some text"
let size: CGSize = placeholderTextView.sizeThatFits(CGSize(width: widthOfActualTextView, height: CGFloat.greatestFiniteMagnitude))

height = size.height

1
投票

我只是浪费了很多时间,所以我提供了一个额外的答案,以便将来拯救其他人。格雷厄姆的回答是90%正确,但缺少一个关键部分:

要使用-boundingRectWithSize:options:获得准确的结果,您必须通过以下选项:

NSStringDrawingUsesLineFragmentOrigin|NSStringDrawingUsesDeviceMetrics|NSStringDrawingUsesFontLeading`

如果省略lineFragmentOrigin,那么你会得到废话;返回的rect将是一行高,并且根本不会考虑您传递给方法的大小。

为什么这么复杂,如此缺乏文件记录超出了我的范围。但是你现在有了。通过这些选项,它将完美地工作(至少在OS X上)。


0
投票

这个页面上没有一个答案对我有用,也没有来自Apple's documentation的古老的Objective-C代码。我最终为UITextView工作的是首先在其上设置textattributedText属性,然后像这样计算所需的大小:

let size = textView.sizeThatFits(CGSize(width: maxWidth, height: CGFloat.max))

完美的工作。 Booyah!

© www.soinside.com 2019 - 2024. All rights reserved.