这一节我们来讲如何修改FBReaderJ为分章加载,从而实现超大Epub书籍的快速加载,文末还有一些关于汉字排版问题的解决方案。修改前后时间对比。
修改前后时间对比
- 修改前 - 全部加载 - 5s
- 修改后 - 分章加载 - 0.5s
一. 整体思路
- 原先的加载策略是:解析完所有章节(xhtml)后再进行显示(EPub格式的书籍里每一章对应一个xhtml页面)
- 修改目标为:先只解析用户上次阅读所在的章节,快速打开显示,随后在后台进行全部章节的加载
二. 详细步骤
- 每次翻页后保存进度至数据库时,计算出当前页所在章节的序号也存入数据库中
- 每次打开加载书籍,从数据库中查出上次保存的章节序号,只进行该章节的加载
- 快速打开书籍后,再进行全部章节的加载,加载完成后,替换两者,完成转换
三. 前提
- 通过分析源代码我们知道,解析xhtml使用C++编写,Android端通过JNI来进行调用,所以这里会用到一些C++的知识
- 关于进度保存,原先是保存当前页首个字的位置(在第几段(解析时每个<p></p>为一段),第几个字,第几个字母(针对英文)),这里我们再加上在第几章
- 解析目录后会得到每一章的开始在整个书籍的位置(第几段(paragraphIndex))
四. 需要解决的问题
1. 如何计算出当前页在第几章节?
这里我们通过得到当前页第一个字的位置,通过for循环与每一章的paragraphIndex进行比较,当大于它时即可判断出当前页所在章节。
2. 如何计算出当前页在该章中的第几段?
(FBReader中为在整本书的第几段)保存时我们要计算出,当前页在该章中的第几段,以保证刚开始只加载一章时可以正确跳转至阅读进度,方法也是得到当前页在整本书的第几段,再减去当前章前面的段数。
3. 加载全部章节后如何保持当前页不变?
(该页在一章和全部章中的位置明显不同) 在进行全部章节加载时,首先要获取当前页所在的位置(在该章中的位置P),在加载完成后,计算出当前章前面所有章的段数,加上P后即可得到正确位置,从而保持当前页不变。
4. 如何修改C++中解析代码
使用JNI调用C++进行解析前,我们要告诉它是解析一章还是全部解析,具体解析哪一章,C++解析时通过得到Java中静态变量的值,判断执行分章加载还是全部加载,以及具体加载哪一章节。
这里看一下修改后的关键代码
-
若为全部加载,则还走原先的方法,通过for循环解析全部章节,可以看到从xhtml的开始(begin),一直循环到了末尾(end)。
-
若为分章加载,则需要修改为只加载当前章节,得到当前章所在xhtml,进行加载。
5.关于笔记的位置保存
之前的策略是只加载一章,当翻到每章末尾时再进行下一章的加载,这样的话笔记的位置则也需要带上所在的章节并进行计算,但这样会使用户每次到末尾都用0.5s加载一下,体验不好。
因此修改策略为加载一章后再加载全部章节,这样的话,笔记的位置就不用修改了,加载完所有章节后再高亮显示出笔记即可。
五. 其它问题
FBReader对中文的处理会出现“狗尾”问题,即汉字无法与右边对齐,如下图,可以看到英文可以与右侧对齐,而中文则无法对齐。-
解决思路:每一行字数不变,但根据每一行的字数以及总宽度,计算出每行每个字之间的间距,通过增大字之间的间距,从而使汉字可以右对齐。
*修改后的界面如下图,可以下载App体验一下效果
这是很早之前写的一篇文章,原文发表在link