For a given NSRange
, I'd like to find a CGRect
in a UILabel
that corresponds to the glyphs of that NSRange
. For example, I'd like to find the CGRect
that contains the word "dog" in the sentence "The quick brown fox jumps over the lazy dog."
The trick is, the UILabel
has multiple lines, and the text is really attributedText
, so it's a bit tough to find the exact position of the string.
The method that I'd like to write on my UILabel
subclass would look something like this:
- (CGRect)rectForSubstringWithRange:(NSRange)range;
Details, for those who are interested:
My goal with this is to be able to create a new UILabel with the exact appearance and position of the UILabel, that I can then animate. I've got the rest figured out, but it's this step in particular that's holding me back at the moment.
What I've done to try and solve the issue so far:
- I'd hoped that with iOS 7, there'd be a bit of Text Kit that would solve this problem, but most every example I've seen with Text Kit focuses on
UITextView
andUITextField
, rather thanUILabel
. - I've seen another question on Stack Overflow here that promises to solve the problem, but the accepted answer is over two years old, and the code doesn't perform well with attributed text.
I'd bet that the right answer to this involves one of the following:
- Using a standard Text Kit method to solve this problem in a single line of code. I'd bet it would involve
NSLayoutManager
andtextContainerForGlyphAtIndex:effectiveRange
- Writing a complex method that breaks the UILabel into lines, and finds the rect of a glyph within a line, likely using Core Text methods. My current best bet is to take apart @mattt's excellent TTTAttributedLabel, which has a method that finds a glyph at a point - if I invert that, and find the point for a glyph, that might work.
Update: Here's a github gist with the three things I've tried so far to solve this issue: https://gist.github.com/bryanjclark/7036101
Building off of Luke Rogers's answer but written in swift:
Swift 2
Example Usage (Swift 2)
Swift 3/4
Example Usage (Swift 3/4)
My suggestion would be to make use of Text Kit. Unfortunately we don't have access to the layout manager that a
UILabel
uses however it might be possible to create a replica of it and use that to get the rect for a range.My suggestion would be to create a
NSTextStorage
object containing the exact same attributed text as is in your label. Next create aNSLayoutManager
and add that to the the text storage object. Finally create aNSTextContainer
with the same size as the label and add that to the layout manager.Now the text storage has the same text as the label and the text container is the same size as the label so we should be able to ask the layout manager we created for a rect for our range using
boundingRectForGlyphRange:inTextContainer:
. Make sure you convert your character range to a glyph range first usingglyphRangeForCharacterRange:actualCharacterRange:
on the layout manager object.All going well that should give you a bounding
CGRect
of the range you specified within the label.I haven't tested this but this would be my approach and by mimicking how the
UILabel
itself works should have a good chance of succeeding.Can you instead base your class on UITextView? If so, check out the UiTextInput protocol methods. See in particular the geometry and hit resting methods.
Another way of doing this, if you have automatic font size adjustment enabled, would be like this:
Translated for Swift 3.
swift 4 solution, will work even for multiline strings, bounds were replaced with intrinsicContentSize