我正在尝试将滚动文本的位置保存在UITextView
中,以便在再次加载ViewController
时可以返回该位置。我的字符串很长,所以我希望用户能够滚动到特定位置,然后稍后再返回该位置。
我正在使用UITextView
。 scrollRangeToVisible函数可以自动滚动文本视图,但是我不知道如何获取用户看到的文本的NSRange。这是解决此问题的最佳方法吗?我尝试使用setContentOffset
函数,但似乎没有任何作用。
感谢您的帮助。谢谢!
答案 0 :(得分:1)
我还没有对它进行彻底的测试,但是我相信以下应该可以。 UITextInput
协议中记录了您需要的API,UITextView
采用了该协议。
您首先需要获取与视图内给定点相对应的UITextPosition
。然后,您可以将此值转换为UTF-16字符偏移量。例如,在这里,每次滚动视图时,我都会打印textView
的可见文本范围(以UTF-16代码单位表示):
func scrollViewDidScroll(_ scrollView: UIScrollView) {
let topLeft = CGPoint(x: textView.bounds.minX, y: textView.bounds.minY)
let bottomRight = CGPoint(x: textView.bounds.maxX, y: textView.bounds.maxY)
guard let topLeftTextPosition = textView.closestPosition(to: topLeft),
let bottomRightTextPosition = textView.closestPosition(to: bottomRight)
else {
return
}
let charOffset = textView.offset(from: textView.beginningOfDocument, to: topLeftTextPosition)
let length = textView.offset(from: topLeftTextPosition, to: bottomRightTextPosition)
let visibleRange = NSRange(location: charOffset, length: length)
print("Visible range: \(visibleRange)")
}
在我的测试中,UITextView
倾向于计算几乎不包含在可见范围内的行(例如,仅一个像素),因此报告的可见范围往往比人类用户大一到两行会说。您可能需要对传入CGPoint
的确切closesPosition(to:)
进行试验,以获得所需的结果。
答案 1 :(得分:1)
在UITextView
上有一个小扩展,它使用了characterRange(at:)
函数。它还添加了一个计算属性,以返回当前可见的文本:
extension UITextView {
private var firstVisibleCharacterPosition: UITextPosition? {
// ⚠️ For some reason `characterRange(at:)` returns nil for points with a low y value.
guard let scrolledPosition = characterRange(at: contentOffset)?.start else {
return beginningOfDocument
}
return scrolledPosition
}
private var lastVisibleCharacterPosition: UITextPosition? {
return characterRange(at: bounds.max)?.end
}
/// The range of the text that is currently displayed within the text view's bounds.
var visibleTextRange: UITextRange? {
guard
let first = firstVisibleCharacterPosition,
let last = lastVisibleCharacterPosition else {
return nil
}
return textRange(from: first, to: last)
}
/// The string that is currently displayed within the text view's bounds.
var visibleText: String? {
guard let visibleTextRange = visibleTextRange else {
return nil
}
return text(in: visibleTextRange)
}
}
我在上面的代码中使用了这些速记属性:
extension CGRect {
var min: CGPoint {
return .init(x: minX, y: minY)
}
var max: CGPoint {
return .init(x: maxX, y: maxY)
}
}