我想在固定宽度的框中绘制一些NSAttributedString,但是在计算绘制它们时会占用的正确高度时遇到问题。到目前为止,我已经尝试过:
- (NSSize) size
,但结果没用(为此目的),因为他们会给出字符串所需的宽度。- (void)drawWithRect:(NSRect)rect options:(NSStringDrawingOptions)options
和选项中的NSStringDrawingUsesLineFragmentOrigin
,就像我在绘图中使用的那样。结果......难以理解;当然不是我想要的。 (正如许多地方所指出的那样,包括this Cocoa-Dev线程)。[[tmpView textStorage] setAttributedString:aString];
[tmpView setHorizontallyResizable:NO];
[tmpView sizeToFit];
当我查询tmpView的框架时,宽度仍然是所希望的,并且高度通常是正确的...直到我得到更长的字符串,当它通常是所需尺寸的一半时。 (似乎没有最大尺寸被击中:一帧将是273.0高(约300太短),另一帧将是478.0(只有60-ish太短))。如果有人管理过这个,我会感激任何指点。
-[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开发人员文档。
正如上面提到的很多人,并根据我的测试。
我在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)
,它给出了相同的结果。
结果是我的预期。
谢谢。
我找到帮助类来查找attributedText的高度和宽度(测试代码)
https://gist.github.com/azimin/aa1a79aefa1cec031152fa63401d2292
在项目中添加以上文件
如何使用
let attribString = AZTextFrameAttributes(attributedString: lbl.attributedText!)
let width : CGFloat = attribString.calculatedTextWidth()
print("width is :: >> \(width)")
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)
答案是使用
- (void)drawWithRect:(NSRect)rect options:(NSStringDrawingOptions)options
但是你传入的rect
应该在你想要无限的维度中有0.0(呃,这是完全合理的)。例子here。
您可能对Jerry Krinock的伟大(仅限OS X)NS(Attributed)String+Geometrics
类别感兴趣,该类别旨在进行各种字符串测量,包括您正在寻找的内容。
我有一个复杂的属性字符串与多个字体,并得到不正确的结果与我尝试的一些上述答案。使用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
}
使用NSAttributedString方法
- (CGRect)boundingRectWithSize:(CGSize)size options:(NSStringDrawingOptions)options context:(NSStringDrawingContext *)context
大小是区域的约束,计算的区域宽度限制为指定的宽度,而高度基于此宽度是灵活的。如果不可用,可以为上下文指定nil。要获得多行文本大小,请使用NSStringDrawingUsesLineFragmentOrigin作为选项。
在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;
}
斯威夫特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
我只是浪费了很多时间,所以我提供了一个额外的答案,以便将来拯救其他人。格雷厄姆的回答是90%正确,但缺少一个关键部分:
要使用-boundingRectWithSize:options:
获得准确的结果,您必须通过以下选项:
NSStringDrawingUsesLineFragmentOrigin|NSStringDrawingUsesDeviceMetrics|NSStringDrawingUsesFontLeading`
如果省略lineFragmentOrigin,那么你会得到废话;返回的rect将是一行高,并且根本不会考虑您传递给方法的大小。
为什么这么复杂,如此缺乏文件记录超出了我的范围。但是你现在有了。通过这些选项,它将完美地工作(至少在OS X上)。
这个页面上没有一个答案对我有用,也没有来自Apple's documentation的古老的Objective-C代码。我最终为UITextView
工作的是首先在其上设置text
或attributedText
属性,然后像这样计算所需的大小:
let size = textView.sizeThatFits(CGSize(width: maxWidth, height: CGFloat.max))
完美的工作。 Booyah!