[Android] TextView 分页功能的实现

摘要: 分页功能是阅读器类软件的基本功能之一, 之前自己写阅读器时第一个遇到的问题就是这个. 也尝试了不少办法来解决这个问题, 现在把其中最容易实现的一个方法记录下来, 也方便大家参考。
本文转载自:https://my.oschina.net/gotax/blog/136860

分页功能是阅读器类软件的基本功能之一, 也是自己之前写阅读器时遇到的第一个问题. 尝试了不少办法才解决, 现在把其中最容易实现的一个方法记录下来, 也方便大家参考.
基本思路如下:

  1. 从文件中读取 8000 个字符至缓冲区
  2. 将表示位置的指针指向缓冲区开头
  3. 让 TextView 显示从指针所指位置开始的内容
  4. 获取 TextView 中的可见字数 n
  5. 将指针向后移动 n 位
  6. 向后翻页时执行 3 ~ 5 步

整体思路很简单, 其中唯一的难点就是第 4 步, 如何获取 TextView 中的可见字数.
我遇到这类问题一般就是两步走, 先文档, 后源码.
所以先去查 Android 文档, 看看 TextView 有没有什么可以利用的函数. 在其中找到一个函数:

getLineBounds(int line, Rect bounds)// 得到指定行的边界

似乎有点用. 只要从第一行开始一行一行往下看, 直到找到超出边界的那一行, 就能知道这个 TextView 能显示多少行了. 或者用 getHeight() / getLineHeight() 也能获取 TextView 的最大显示行数. 但由于并不知道每行的字数, 所以还是算不出来一页到底有多少字.

后来又尝试了许多其他方法, 也在提问区问过. 结果只得到了一个建议, 就是自己写个 View. 整个 View 都由自己实现的话, 的确能很方便地控制所有细节, 但随之而来的麻烦就是, 所有的细节都得自己实现. 比如我的断行, 和布局自适应这两点处理得就没原生的 TextView 那么好, 只能说勉强能用. 更别提超链接这类的东西了, 要想全部实现还真不是一时半会能搞定的.

既然查文档无果, 那就只能去看源码了. 不看不知道, 这不起眼的 Textview 源码居然有近 9000 行, 顿时有点犯晕. 不过我的目标只有一个, 搞清楚 TextView 是怎么排版的. 所以直接看 ``onDraw(Canvas canvas) 函数, 在其中找到这么一行:

layout.draw(canvas, highlight, mHighlightPaint, cursorOffsetVertical);

由此可以推断 TextView 排版及绘制文字靠的就是这个 layout, 所以立刻到文档中找 Layout, 这次终于在其中找到了几个有用的函数(就是那些 getLine* 函数), 最有用的是这两个:

getLineForVertical(int vertical)// 根据纵坐标得到对应的行号

getLineEnd(int line)// 返回指定行中最后一个字在整个字符串中的位置

所以我们只要先计算出最下面一行是第几行, 然后再算出这行最后一个字是第几个字就行了.

先算行号:

public int getLineNum() {
    Layout layout = getLayout();
    int topOfLastLine = getHeight() - getPaddingTop() - getPaddingBottom() - getLineHeight();
    return layout.getLineForVertical(topOfLastLine);
} ```

再算字数:

public int getCharNum() {
return getLayout().getLineEnd(getLineNum());
}

这样我们就能得到 TextView 在本页所显示的字数了.

    // 构造函数略...
    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        super.onLayout(changed, left, top, right, bottom);
        resize();
    }

    /**
     * 去除当前页无法显示的字
     * @return 去掉的字数
     */
    public int resize() {
        CharSequence oldContent = getText();
        CharSequence newContent = oldContent.subSequence(0, getCharNum());
        setText(newContent);
        return oldContent.length() - newContent.length();
    }

    /**
     * 获取当前页总字数
     */
    public int getCharNum() {
        return getLayout().getLineEnd(getLineNum());
    }

    /**
     * 获取当前页总行数
     */
    public int getLineNum() {
        Layout layout = getLayout();
        int topOfLastLine = getHeight() - getPaddingTop() - getPaddingBottom() - getLineHeight();
        return layout.getLineForVertical(topOfLastLine);
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 176,034评论 25 709
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 14,479评论 4 61
  • 在去南方的火车上,作者注意到她旁边的那位女子始终低着头看着像是日记之类的记事本。或许她注意到了我的目光,转过头来笑...
    我姓胡_阅读 1,774评论 0 0
  • <想你一次是一粒沙是一滴水> 究竟想了你多少次 三毛说撒哈拉有多少粒沙 太平洋就有多少滴水 <写诗> 我喜欢 一边...
    凡高潇湘花子阅读 2,144评论 10 1
  • 今天是2017年1月29日星期天。 又是新的一年了,可是这又怎们样呢,能怎么样呢,还不是白天一样黑,黑夜一样光怪陆...
    七月我想啃西瓜阅读 3,101评论 0 1

友情链接更多精彩内容