制作安卓PDF阅读器:九、实现缩略图网格和全文搜索

2020-12-05  本文已影响0人  天下第九九八十一

实现缩略图卷轴后,一下子就爆发出很多的可能性。这些方向主要有:缩略图网格、页面跳转的历史记录、全文搜索、以及书签。

一、缩略图网格

mViewPager.setLayoutManager(new GridLayoutManager(a, 6));

这样就完成6列网格的显示了。GridLayoutManager 扩展自 LinearLayoutManager,所以不会出问题,但具体还需调整。

网格切换 横向NumberPicker,可以用键盘输入数字。

二、实现PDF全文搜索

2.1 搜索界面

通过改写recyclerview的适配器,显示全部页面的子集,作为搜索结果的基本展示。

底部数据就一个 Arraylist,存放具体页面的索引。所以也可以使用SparseIntArray。

实现 ResultsProvider 接口将底部数据隐藏起来。

运用广度优先的搜索策略,即优先搜索哪些页面有匹配,而不是全部匹配的具体位置。

2.2 PDF 全文搜索

API : FPDFText_FindStart、FPDFText_FindNext。暂时这两个就够,一页页地调用两个API各一次进行搜索。

需要用 FPDF_LoadPage、FPDFText_LoadPage分别加载每一页及其文本对象。测试发现,七百多页的内存中的Gpu pro 1,加载全部页面耗时2.7秒,加载全部页面的文本对象耗时4.4-2.7秒。对每一页面发起并关闭一次搜索的时间可忽略不计。

也就是说什么搜没开始搜索,就有一个4.4秒左右的基本耗时。

JNI_FUNC(void, PdfiumCore, nativeFindAll)(JNI_ARGS, jlong docPtr, jint pages) {
    DocumentFile* docFile = (DocumentFile*)docPtr;
    // 测试搜索功能。 
    const unsigned short nope[] = { 'n', 'o', 'p', 'e', '\0' }; // 模仿测试文件 fpdftext_embeddertest.cpp
    FPDF_DOCUMENT pdfDoc = docFile->pdfDocument;
    if(pdfDoc){
        for(int i=0;i<pages;i++) { // 对每一页面
            FPDF_PAGE page = FPDF_LoadPage(pdfDoc, i); //加载页面
            if(page) {
                FPDF_TEXTPAGE text = FPDFText_LoadPage(page); //加载文本
                if(text) {
                    // 发起搜索
                    FPDF_SCHHANDLE findHandle = FPDFText_FindStart(text, nope, 0, 0); 
                     // 关闭搜索
                    FPDFText_FindClose(findHandle);

                    FPDFText_ClosePage(text); //卸载文本
                }
                FPDF_ClosePage(page); //卸载页面
            }
        }
    }
}

进一步测试,在发起搜索和关闭搜索之间正式搜索一次。耗时4.5秒,还是很快的。

FPDFText_FindNext(findHandle); // 正式搜索

这样一来就可以放心使用API了。接下来将有匹配的页面置入ArrayList里面,最后传给上面说的缩略图页面子集适配器就可以了!

附从jstring获取可以传给FPDFText_FindStart的keyStr:

const unsigned short * keyStr = env->GetStringChars(key, 0);

如果选择缓存全部页面对象而不关闭,那么第二次搜索将快很多(200到300毫秒);但是会占用内存空间,七百多页共占0.2GB左右。

事实也证明页面及其文本对象的加载是主要耗时过程。相比之下,jni调用、jstring转换等的消耗不值一提。


2.3 绘制高亮

又是与绘制选区差不多呗。拿到基本的搜索结果后,渲染时进一步循环调用 FPDFText_FindNext 搜索直到这一页返回false。循环之内,用FPDFText_GetSchResultIndex、FPDFText_GetSchCount获取一页内所有匹配的字符索引。然后和之前一样用FPDFText_CountRects 和 FPDFText_GetRect获得一些需要绘制的矩形。

这个功能是 JNI 调用最为密集的,因此简单测试了一下 JNI 调用的效率,发现在native层调用java方法要比正向调用慢一些,不知是否是测试方法的问题。

然后是绘制方面,之前说若将选区方框绘制在后面需要分好几层,很麻烦。实际则简单得多,只需用paint.setXfermode(PorterDuff.Mode.DARKEN)就可以达成类似的效果,神奇。

结果如图所示,使用YPWaveView(动态波浪视图)指示搜索进度,效果相当不错。

……

上一篇 下一篇

猜你喜欢

热点阅读