Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Get line information from UITextView and NSLayoutManager

In order to support the UIAccessibilityReadingContent protocol, I need my UITextView to answer me questions about its lines. These are the methods of the protocol that I need to implement:

  • accessibilityLineNumberForPoint: <- Provided a coordinate, return a line number
  • accessibilityContentForLineNumber: <- Return the text of a given line
  • accessibilityFrameForLineNumber: <- Given a line number, return its frame
  • accessibilityPageContent <- The entire text content. That I have. :)

I figure that NSLayoutManager can help me, but I'm not that experienced with it. I've figured some of it out (I think), but still need some help.

Apple has some sample code (here) that can get me the number of lines in the text view:

NSLayoutManager *layoutManager = [textView layoutManager];
unsigned numberOfLines, index, numberOfGlyphs =
        [layoutManager numberOfGlyphs];
NSRange lineRange;
for (numberOfLines = 0, index = 0; index < numberOfGlyphs; numberOfLines++){
    (void) [layoutManager lineFragmentRectForGlyphAtIndex:index
            effectiveRange:&lineRange];
    index = NSMaxRange(lineRange);
}

I figure that with lineRangeabove, I can calculate the line rects using this method on NSLayoutManager:

- (NSRect)boundingRectForGlyphRange:(NSRange)glyphRange inTextContainer:(NSTextContainer *)container

And given lineRanges I should be able to calculate the line number for a point using (by finding the lineRange that contains the glyph index:

- (NSUInteger)glyphIndexForPoint:(CGPoint)point inTextContainer:(NSTextContainer *)container fractionOfDistanceThroughGlyph:(CGFloat *)partialFraction

So what remains is, how do I get the content of a line (as an NSString), given a line number?

like image 395
Mikkel Selsøe Avatar asked Apr 02 '14 10:04

Mikkel Selsøe


1 Answers

thinking in an easy solution, here is a small code that reproduce what you need

- (void)analyse:(UITextView *)textView
{
    NSLayoutManager *layoutManager = [textView layoutManager];
    NSString *string = textView.text;
    unsigned numberOfLines, index, stringLength = [string length];

    NSMutableArray *ranges = [NSMutableArray new];
    NSMutableArray *frames = [NSMutableArray new];
    for (index = 0, numberOfLines = 0; index < stringLength; numberOfLines++)
    {
        NSRange tmprange;
        NSRange range = [string lineRangeForRange:NSMakeRange(index, 0)];
        CGRect rect = [layoutManager lineFragmentRectForGlyphAtIndex:index
                                           effectiveRange:&tmprange];

        [ranges addObject:[NSValue valueWithRange:range]];

        [frames addObject:[NSValue valueWithCGRect:rect]];
        index = NSMaxRange(tmpRange);
    }
    self.ranges = ranges;
    self.frames = frames;
    self.numberOfLines = numberOfLines;
}

Please take a look of the properties:

self.ranges = ranges;
self.frames = frames;
self.numberOfLines = numberOfLines;

You can have the following in your class to create this properties:

@property (nonatomic) NSInteger numberOfLines;
@property (strong, nonatomic) NSArray *ranges;
@property (strong, nonatomic) NSArray *frames;

I suggest you to add the analyse call inside the following delegate:

- (void)textViewDidChange:(UITextView *)textView

There you can for example after analyse get the frame of the 2nd line just doing: self.frames[1]

Or getting the text of the second line doing: [textView.text substringWithRange:[self.ranges[1] rangeValue]]

For example like this:

if (self.numberOfLines > 1)
{
    NSRange range = [self.ranges[1] rangeValue];
    NSLog(@"2nd line = %@", [textView.text substringWithRange:range]);
    NSLog(@"2nd line frame = %@", self.frames[1]);
}

Having all the frames in self.frames I think you can easily do the other thing, guess the line number using a coordinate.

like image 88
southfox Avatar answered Nov 12 '22 20:11

southfox