在UITextView中滚动后,获取可见文本的NSRange

时间:2018-06-29 18:16:49

标签: ios swift uitextview nsrange

我正在尝试将滚动文本的位置保存在UITextView中,以便在再次加载ViewController时可以返回该位置。我的字符串很长,所以我希望用户能够滚动到特定位置,然后稍后再返回该位置。

我正在使用UITextView。 scrollRangeToVisible函数可以自动滚动文本视图,但是我不知道如何获取用户看到的文本的NSRange。这是解决此问题的最佳方法吗?我尝试使用setContentOffset函数,但似乎没有任何作用。

感谢您的帮助。谢谢!

2 个答案:

答案 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)
    }

}