Ios - как найти видимый диапазон текста в UITextView?
как найти, какой текст отображается в прокручиваемом, неэдоптируемом UITextView?
например, мне может потребоваться показать следующий абзац, затем я хочу найти текущий видимый диапазон текста и использовать его для вычисления соответствующего диапазона и использовать scrollRangeToVisible:
для прокрутки текстового вида
Ответы
Ответ 1
То, как я буду делать это, - это вычислить все размеры каждого абзаца. С sizeWithFont: constrainedToSize: lineBreakMode:
вы сможете определить, какой абзац видим, из [textView contentOffset].
для прокрутки, не используйте scrollRangeToVisible, просто используйте setContentOffset: параметр CGPoint y для этого должен быть либо суммой всех размеров высоты для следующего абзаца, либо просто добавить textView.frame.size.height, если это ближе к началу следующего абзаца.
Это имеет смысл?
в ответ на комментарий requst code ниже (untested):
CGFloat paragraphOffset[MAX_PARAGRAPHS];
CGSize constraint = CGSizeMake(widthOfTextView, 999999 /*arbitrarily large number*/);
NSInteger paragraphNo = 0;
CGFloat offset = 0;
for (NSString* paragraph in paragraphs) {
paragraphOffset[paragraphNo++] = offset;
CGSize paragraphSize = [paragraph sizeWithFont:textView.font constrainedToSize:constraint lineBreakMode:UILineBreakModeWordWrap];
offset += paragraphSize.height;
}
// find visible paragraph
NSInteger visibleParagraph = 0;
while (paragraphOffset[visibleParagraph++] < textView.contentOffset.y);
// scroll to paragraph 6
[textView setContentOffset:CGPointMake(0, paragraphOffset[6]) animated:YES];
Ответ 2
Я нашел другое решение здесь.
Это лучший способ решить эту проблему в моих глазах.
fooobar.com/questions/321245/...
Так как UITextView является подклассом UIScrollView, его свойство bounds отражает видимую часть его системы координат. Итак, что-то вроде этого должно работать:
-(NSRange)visibleRangeOfTextView:(UITextView *)textView {
CGRect bounds = textView.bounds;
UITextPosition *start = [textView characterRangeAtPoint:bounds.origin].start;
UITextPosition *end = [textView characterRangeAtPoint:CGPointMake(CGRectGetMaxX(bounds), CGRectGetMaxY(bounds))].end;
return NSMakeRange([textView offsetFromPosition:textView.beginningOfDocument toPosition:start],
[textView offsetFromPosition:start toPosition:end]);
}
Это предполагает макет текста сверху вниз, слева направо. Если вы хотите, чтобы он работал для других направлений макета, вам придется больше работать.:)
Ответ 3
Если вы хотите использовать решение Swift, я использую это:
public extension UITextView {
public var visibleRange: NSRange? {
if let start = closestPositionToPoint(contentOffset) {
if let end = characterRangeAtPoint(CGPointMake(contentOffset.x + CGRectGetMaxX(bounds), contentOffset.y + CGRectGetMaxY(bounds)))?.end {
return NSMakeRange(offsetFromPosition(beginningOfDocument, toPosition: start), offsetFromPosition(start, toPosition: end))
}
}
return nil
}
}
Ответ 4
Один из вариантов - использовать UIWebView вместо UITextView. Затем вы можете использовать якоря и javascript для прокрутки до соответствующих мест в тексте. Вы можете, возможно, вставить анкеры программно в начале каждого абзаца, чтобы сделать это проще.
Ответ 5
Swift 3.0/3.1 решение, основанное на ответе "Noodle of Death".
public extension UITextView {
public var visibleRange: NSRange? {
if let start = closestPosition(to: contentOffset) {
if let end = characterRange(at: CGPoint(x: contentOffset.x + bounds.maxX, y: contentOffset.y + bounds.maxY))?.end {
return NSMakeRange(offset(from: beginningOfDocument, to: start), offset(from: start, to: end))
}
}
return nil
}
}