NSLayoutManager boundingRectForGlyphRange 关闭一些点
NSLayoutManager boundingRectForGlyphRange off by some Points
我想 "highlight" UITextView 中的特定单词,而不是纯色(通过 NSAttributedString
可以实现),而是使用渐变和其他一些奇特的效果。
因此,我决定有必要手动创建一个视图并使用给定文本的边界矩形覆盖(或底层)它。
令我惊讶的是,事实证明这非常简单。示例在 UIViewController 中,txtView
是 UITextView
连接为 IBOutlet
:
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
// Converting to NSString as I need a NSRange for glphyRangeForCharacterRange
// (a normal String in Swift returns a Range)
let charRange = (txtView.text as NSString).rangeOfString("dolor sit er elit")
assert(charRange.location != NSNotFound, "Character Range could not be found.")
let glyphRange = txtView.layoutManager.glyphRangeForCharacterRange(charRange,
actualCharacterRange: nil)
let glyphContainer = txtView.layoutManager.textContainerForGlyphAtIndex(glyphRange.location, effectiveRange: nil)
let glyphRect = txtView.layoutManager.boundingRectForGlyphRange(glyphRange,
inTextContainer: glyphContainer!)
let highlightView = UIView(frame: glyphRect)
highlightView.alpha = 0.3
highlightView.backgroundColor = UIColor.redColor()
txtView.addSubview(highlightView)
}
遗憾的是,这导致重叠视图(在下面的屏幕截图中以红色显示)偏离了几分。
它似乎总是有相同的偏差。这意味着我可能会硬编码,但这从来都不是一个好主意。在 Simulator 中使用各种设备进行测试,并在 iPhone 6 上使用 iOS 8.1.3.
进行测试
我还尝试了其他变体(通过创建我自己的 NSTextContainer 等),受到 the question on how to get a CGRect for a substring and "boundingRectForGlyphRange does not always work for all strings"
的启发
有什么想法(除了 resorting to CoreText)吗?
尝试通过文本视图的 textContainerInset 调整字形矩形:
var glyphRect = txtView.layoutManager.boundingRectForGlyphRange(glyphRange,
inTextContainer: glyphContainer!)
glyphRect.origin.y += txtView.textContainerInset.top
这应该可以解决问题![=11=]
我想 "highlight" UITextView 中的特定单词,而不是纯色(通过 NSAttributedString
可以实现),而是使用渐变和其他一些奇特的效果。
因此,我决定有必要手动创建一个视图并使用给定文本的边界矩形覆盖(或底层)它。
令我惊讶的是,事实证明这非常简单。示例在 UIViewController 中,txtView
是 UITextView
连接为 IBOutlet
:
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
// Converting to NSString as I need a NSRange for glphyRangeForCharacterRange
// (a normal String in Swift returns a Range)
let charRange = (txtView.text as NSString).rangeOfString("dolor sit er elit")
assert(charRange.location != NSNotFound, "Character Range could not be found.")
let glyphRange = txtView.layoutManager.glyphRangeForCharacterRange(charRange,
actualCharacterRange: nil)
let glyphContainer = txtView.layoutManager.textContainerForGlyphAtIndex(glyphRange.location, effectiveRange: nil)
let glyphRect = txtView.layoutManager.boundingRectForGlyphRange(glyphRange,
inTextContainer: glyphContainer!)
let highlightView = UIView(frame: glyphRect)
highlightView.alpha = 0.3
highlightView.backgroundColor = UIColor.redColor()
txtView.addSubview(highlightView)
}
遗憾的是,这导致重叠视图(在下面的屏幕截图中以红色显示)偏离了几分。
它似乎总是有相同的偏差。这意味着我可能会硬编码,但这从来都不是一个好主意。在 Simulator 中使用各种设备进行测试,并在 iPhone 6 上使用 iOS 8.1.3.
进行测试我还尝试了其他变体(通过创建我自己的 NSTextContainer 等),受到 the question on how to get a CGRect for a substring and "boundingRectForGlyphRange does not always work for all strings"
的启发有什么想法(除了 resorting to CoreText)吗?
尝试通过文本视图的 textContainerInset 调整字形矩形:
var glyphRect = txtView.layoutManager.boundingRectForGlyphRange(glyphRange,
inTextContainer: glyphContainer!)
glyphRect.origin.y += txtView.textContainerInset.top
这应该可以解决问题![=11=]