mirror of
https://github.com/geometer/FBReaderJ.git
synced 2025-10-03 17:59:33 +02:00
synchronization with native branch (readModel in JavaNativeFormatPlugin)
This commit is contained in:
parent
739a915724
commit
8452c9292b
5 changed files with 288 additions and 1 deletions
|
@ -21,7 +21,9 @@
|
||||||
|
|
||||||
#include <AndroidUtil.h>
|
#include <AndroidUtil.h>
|
||||||
|
|
||||||
|
#include "fbreader/src/bookmodel/BookModel.h"
|
||||||
#include "fbreader/src/formats/FormatPlugin.h"
|
#include "fbreader/src/formats/FormatPlugin.h"
|
||||||
|
#include "fbreader/src/library/Library.h"
|
||||||
#include "fbreader/src/library/Author.h"
|
#include "fbreader/src/library/Author.h"
|
||||||
#include "fbreader/src/library/Book.h"
|
#include "fbreader/src/library/Book.h"
|
||||||
#include "fbreader/src/library/Tag.h"
|
#include "fbreader/src/library/Tag.h"
|
||||||
|
@ -101,9 +103,177 @@ JNIEXPORT jboolean JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPl
|
||||||
return JNI_FALSE;
|
return JNI_FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool initBookModel(JNIEnv *env, jobject javaModel, BookModel &model) {
|
||||||
|
shared_ptr<ZLImageMapWriter> imageMapWriter = model.imageMapWriter();
|
||||||
|
|
||||||
|
env->PushLocalFrame(16);
|
||||||
|
|
||||||
|
jobjectArray ids = AndroidUtil::createStringArray(env, imageMapWriter->identifiers());
|
||||||
|
jintArray indices = AndroidUtil::createIntArray(env, imageMapWriter->indices());
|
||||||
|
jintArray offsets = AndroidUtil::createIntArray(env, imageMapWriter->offsets());
|
||||||
|
jstring imageDirectoryName = env->NewStringUTF(imageMapWriter->allocator().directoryName().c_str());
|
||||||
|
jstring imageFileExtension = env->NewStringUTF(imageMapWriter->allocator().fileExtension().c_str());
|
||||||
|
jint imageBlocksNumber = imageMapWriter->allocator().blocksNumber();
|
||||||
|
env->CallVoidMethod(javaModel, AndroidUtil::MID_NativeBookModel_initImageMap,
|
||||||
|
ids, indices, offsets, imageDirectoryName, imageFileExtension, imageBlocksNumber);
|
||||||
|
env->PopLocalFrame(0);
|
||||||
|
return !env->ExceptionCheck();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool initInternalHyperlinks(JNIEnv *env, jobject javaModel, BookModel &model) {
|
||||||
|
ZLCachedMemoryAllocator allocator(131072, Library::Instance().cacheDirectory(), "nlinks");
|
||||||
|
|
||||||
|
ZLUnicodeUtil::Ucs2String ucs2id;
|
||||||
|
ZLUnicodeUtil::Ucs2String ucs2modelId;
|
||||||
|
|
||||||
|
const std::map<std::string,BookModel::Label> &links = model.internalHyperlinks();
|
||||||
|
std::map<std::string,BookModel::Label>::const_iterator it = links.begin();
|
||||||
|
for (; it != links.end(); ++it) {
|
||||||
|
const std::string &id = it->first;
|
||||||
|
const BookModel::Label &label = it->second;
|
||||||
|
if (label.Model.isNull()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
ZLUnicodeUtil::utf8ToUcs2(ucs2id, id);
|
||||||
|
ZLUnicodeUtil::utf8ToUcs2(ucs2modelId, label.Model->id());
|
||||||
|
const size_t idLen = ucs2id.size() * 2;
|
||||||
|
const size_t modelIdLen = ucs2modelId.size() * 2;
|
||||||
|
|
||||||
|
char *ptr = allocator.allocate(idLen + modelIdLen + 8);
|
||||||
|
ZLCachedMemoryAllocator::writeUInt16(ptr, ucs2id.size());
|
||||||
|
ptr += 2;
|
||||||
|
memcpy(ptr, &ucs2id.front(), idLen);
|
||||||
|
ptr += idLen;
|
||||||
|
ZLCachedMemoryAllocator::writeUInt16(ptr, ucs2modelId.size());
|
||||||
|
ptr += 2;
|
||||||
|
memcpy(ptr, &ucs2modelId.front(), modelIdLen);
|
||||||
|
ptr += modelIdLen;
|
||||||
|
ZLCachedMemoryAllocator::writeUInt32(ptr, label.ParagraphNumber);
|
||||||
|
}
|
||||||
|
allocator.flush();
|
||||||
|
|
||||||
|
jstring linksDirectoryName = env->NewStringUTF(allocator.directoryName().c_str());
|
||||||
|
jstring linksFileExtension = env->NewStringUTF(allocator.fileExtension().c_str());
|
||||||
|
jint linksBlocksNumber = allocator.blocksNumber();
|
||||||
|
env->CallVoidMethod(javaModel, AndroidUtil::MID_NativeBookModel_initInternalHyperlinks,
|
||||||
|
linksDirectoryName, linksFileExtension, linksBlocksNumber);
|
||||||
|
env->DeleteLocalRef(linksDirectoryName);
|
||||||
|
env->DeleteLocalRef(linksFileExtension);
|
||||||
|
return !env->ExceptionCheck();
|
||||||
|
}
|
||||||
|
|
||||||
|
static jobject createTextModel(JNIEnv *env, jobject javaModel, ZLTextModel &model) {
|
||||||
|
env->PushLocalFrame(16);
|
||||||
|
|
||||||
|
jstring id = AndroidUtil::createJavaString(env, model.id());
|
||||||
|
jstring language = AndroidUtil::createJavaString(env, model.language());
|
||||||
|
jint paragraphsNumber = model.paragraphsNumber();
|
||||||
|
|
||||||
|
const size_t arraysSize = model.startEntryIndices().size();
|
||||||
|
jintArray entryIndices = env->NewIntArray(arraysSize);
|
||||||
|
jintArray entryOffsets = env->NewIntArray(arraysSize);
|
||||||
|
jintArray paragraphLenghts = env->NewIntArray(arraysSize);
|
||||||
|
jintArray textSizes = env->NewIntArray(arraysSize);
|
||||||
|
jbyteArray paragraphKinds = env->NewByteArray(arraysSize);
|
||||||
|
env->SetIntArrayRegion(entryIndices, 0, arraysSize, &model.startEntryIndices().front());
|
||||||
|
env->SetIntArrayRegion(entryOffsets, 0, arraysSize, &model.startEntryOffsets().front());
|
||||||
|
env->SetIntArrayRegion(paragraphLenghts, 0, arraysSize, &model.paragraphLengths().front());
|
||||||
|
env->SetIntArrayRegion(textSizes, 0, arraysSize, &model.textSizes().front());
|
||||||
|
env->SetByteArrayRegion(paragraphKinds, 0, arraysSize, &model.paragraphKinds().front());
|
||||||
|
|
||||||
|
jstring directoryName = env->NewStringUTF(model.allocator().directoryName().c_str());
|
||||||
|
jstring fileExtension = env->NewStringUTF(model.allocator().fileExtension().c_str());
|
||||||
|
jint blocksNumber = (jint) model.allocator().blocksNumber();
|
||||||
|
|
||||||
|
jobject textModel = env->CallObjectMethod(javaModel, AndroidUtil::MID_NativeBookModel_createTextModel,
|
||||||
|
id, language,
|
||||||
|
paragraphsNumber, entryIndices, entryOffsets,
|
||||||
|
paragraphLenghts, textSizes, paragraphKinds,
|
||||||
|
directoryName, fileExtension, blocksNumber);
|
||||||
|
|
||||||
|
if (env->ExceptionCheck()) {
|
||||||
|
textModel = 0;
|
||||||
|
}
|
||||||
|
return env->PopLocalFrame(textModel);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool initTOC(JNIEnv *env, jobject javaModel, BookModel &model) {
|
||||||
|
ContentsModel &contentsModel = (ContentsModel&)*model.contentsModel();
|
||||||
|
|
||||||
|
jobject javaTextModel = createTextModel(env, javaModel, contentsModel);
|
||||||
|
if (javaTextModel == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<jint> childrenNumbers;
|
||||||
|
std::vector<jint> referenceNumbers;
|
||||||
|
const size_t size = contentsModel.paragraphsNumber();
|
||||||
|
childrenNumbers.reserve(size);
|
||||||
|
referenceNumbers.reserve(size);
|
||||||
|
for (size_t pos = 0; pos < size; ++pos) {
|
||||||
|
ZLTextTreeParagraph *par = (ZLTextTreeParagraph*)contentsModel[pos];
|
||||||
|
childrenNumbers.push_back(par->children().size());
|
||||||
|
referenceNumbers.push_back(contentsModel.reference(par));
|
||||||
|
}
|
||||||
|
jintArray javaChildrenNumbers = AndroidUtil::createIntArray(env, childrenNumbers);
|
||||||
|
jintArray javaReferenceNumbers = AndroidUtil::createIntArray(env, referenceNumbers);
|
||||||
|
|
||||||
|
env->CallVoidMethod(javaModel, AndroidUtil::MID_NativeBookModel_initTOC,
|
||||||
|
javaTextModel, javaChildrenNumbers, javaReferenceNumbers);
|
||||||
|
|
||||||
|
env->DeleteLocalRef(javaTextModel);
|
||||||
|
env->DeleteLocalRef(javaChildrenNumbers);
|
||||||
|
env->DeleteLocalRef(javaReferenceNumbers);
|
||||||
|
return !env->ExceptionCheck();
|
||||||
|
}
|
||||||
|
|
||||||
extern "C"
|
extern "C"
|
||||||
JNIEXPORT jboolean JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPlugin_readModel(JNIEnv* env, jobject thiz, jobject javaModel) {
|
JNIEXPORT jboolean JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPlugin_readModel(JNIEnv* env, jobject thiz, jobject javaModel) {
|
||||||
return JNI_FALSE;
|
shared_ptr<FormatPlugin> plugin = findCppPlugin(env, thiz);
|
||||||
|
if (plugin.isNull()) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
jobject javaBook = env->GetObjectField(javaModel, AndroidUtil::FID_NativeBookModel_Book);
|
||||||
|
|
||||||
|
shared_ptr<Book> book = Book::loadFromJavaBook(env, javaBook);
|
||||||
|
shared_ptr<BookModel> model = new BookModel(book);
|
||||||
|
if (!plugin->readModel(*model)) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
model->flush();
|
||||||
|
|
||||||
|
if (!initBookModel(env, javaModel, *model) ||
|
||||||
|
!initInternalHyperlinks(env, javaModel, *model) ||
|
||||||
|
!initTOC(env, javaModel, *model)) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
shared_ptr<ZLTextModel> textModel = model->bookTextModel();
|
||||||
|
jobject javaTextModel = createTextModel(env, javaModel, *textModel);
|
||||||
|
if (javaTextModel == 0) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
env->CallVoidMethod(javaModel, AndroidUtil::MID_NativeBookModel_setBookTextModel, javaTextModel);
|
||||||
|
if (env->ExceptionCheck()) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
env->DeleteLocalRef(javaTextModel);
|
||||||
|
|
||||||
|
const std::map<std::string,shared_ptr<ZLTextModel> > &footnotes = model->footnotes();
|
||||||
|
std::map<std::string,shared_ptr<ZLTextModel> >::const_iterator it = footnotes.begin();
|
||||||
|
for (; it != footnotes.end(); ++it) {
|
||||||
|
jobject javaFootnoteModel = createTextModel(env, javaModel, *it->second);
|
||||||
|
if (javaFootnoteModel == 0) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
env->CallVoidMethod(javaModel, AndroidUtil::MID_NativeBookModel_setFootnoteModel, javaFootnoteModel);
|
||||||
|
if (env->ExceptionCheck()) {
|
||||||
|
return JNI_FALSE;
|
||||||
|
}
|
||||||
|
env->DeleteLocalRef(javaFootnoteModel);
|
||||||
|
}
|
||||||
|
return JNI_TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C"
|
extern "C"
|
||||||
|
|
|
@ -33,6 +33,7 @@ const char * const AndroidUtil::Class_Paths = "org/geometerplus/fbreader/Paths";
|
||||||
const char * const AndroidUtil::Class_ZLFile = "org/geometerplus/zlibrary/core/filesystem/ZLFile";
|
const char * const AndroidUtil::Class_ZLFile = "org/geometerplus/zlibrary/core/filesystem/ZLFile";
|
||||||
const char * const AndroidUtil::Class_Book = "org/geometerplus/fbreader/library/Book";
|
const char * const AndroidUtil::Class_Book = "org/geometerplus/fbreader/library/Book";
|
||||||
const char * const AndroidUtil::Class_Tag = "org/geometerplus/fbreader/library/Tag";
|
const char * const AndroidUtil::Class_Tag = "org/geometerplus/fbreader/library/Tag";
|
||||||
|
const char * const AndroidUtil::Class_NativeBookModel = "org/geometerplus/fbreader/bookmodel/NativeBookModel";
|
||||||
|
|
||||||
jobject AndroidUtil::OBJECT_java_lang_System_err;
|
jobject AndroidUtil::OBJECT_java_lang_System_err;
|
||||||
|
|
||||||
|
@ -78,6 +79,14 @@ jmethodID AndroidUtil::MID_Book_addTag;
|
||||||
|
|
||||||
jmethodID AndroidUtil::SMID_Tag_getTag;
|
jmethodID AndroidUtil::SMID_Tag_getTag;
|
||||||
|
|
||||||
|
jfieldID AndroidUtil::FID_NativeBookModel_Book;
|
||||||
|
jmethodID AndroidUtil::MID_NativeBookModel_initImageMap;
|
||||||
|
jmethodID AndroidUtil::MID_NativeBookModel_initInternalHyperlinks;
|
||||||
|
jmethodID AndroidUtil::MID_NativeBookModel_initTOC;
|
||||||
|
jmethodID AndroidUtil::MID_NativeBookModel_createTextModel;
|
||||||
|
jmethodID AndroidUtil::MID_NativeBookModel_setBookTextModel;
|
||||||
|
jmethodID AndroidUtil::MID_NativeBookModel_setFootnoteModel;
|
||||||
|
|
||||||
JNIEnv *AndroidUtil::getEnv() {
|
JNIEnv *AndroidUtil::getEnv() {
|
||||||
JNIEnv *env;
|
JNIEnv *env;
|
||||||
ourJavaVM->GetEnv((void **)&env, JNI_VERSION_1_2);
|
ourJavaVM->GetEnv((void **)&env, JNI_VERSION_1_2);
|
||||||
|
@ -164,6 +173,16 @@ bool AndroidUtil::init(JavaVM* jvm) {
|
||||||
CHECK_NULL( SMID_Tag_getTag = env->GetStaticMethodID(cls, "getTag", "(Lorg/geometerplus/fbreader/library/Tag;Ljava/lang/String;)Lorg/geometerplus/fbreader/library/Tag;") );
|
CHECK_NULL( SMID_Tag_getTag = env->GetStaticMethodID(cls, "getTag", "(Lorg/geometerplus/fbreader/library/Tag;Ljava/lang/String;)Lorg/geometerplus/fbreader/library/Tag;") );
|
||||||
env->DeleteLocalRef(cls);
|
env->DeleteLocalRef(cls);
|
||||||
|
|
||||||
|
CHECK_NULL( cls = env->FindClass(Class_NativeBookModel) );
|
||||||
|
CHECK_NULL( FID_NativeBookModel_Book = env->GetFieldID(cls, "Book", "Lorg/geometerplus/fbreader/library/Book;") );
|
||||||
|
CHECK_NULL( MID_NativeBookModel_initImageMap = env->GetMethodID(cls, "initImageMap", "([Ljava/lang/String;[I[ILjava/lang/String;Ljava/lang/String;I)V") );
|
||||||
|
CHECK_NULL( MID_NativeBookModel_initInternalHyperlinks = env->GetMethodID(cls, "initInternalHyperlinks", "(Ljava/lang/String;Ljava/lang/String;I)V") );
|
||||||
|
CHECK_NULL( MID_NativeBookModel_initTOC = env->GetMethodID(cls, "initTOC", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;[I[I)V") );
|
||||||
|
CHECK_NULL( MID_NativeBookModel_createTextModel = env->GetMethodID(cls, "createTextModel", "(Ljava/lang/String;Ljava/lang/String;I[I[I[I[I[BLjava/lang/String;Ljava/lang/String;I)Lorg/geometerplus/zlibrary/text/model/ZLTextModel;") );
|
||||||
|
CHECK_NULL( MID_NativeBookModel_setBookTextModel = env->GetMethodID(cls, "setBookTextModel", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;)V") );
|
||||||
|
CHECK_NULL( MID_NativeBookModel_setFootnoteModel = env->GetMethodID(cls, "setFootnoteModel", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;)V") );
|
||||||
|
env->DeleteLocalRef(cls);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -214,6 +233,35 @@ std::string AndroidUtil::convertNonUtfString(const std::string &str) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
jintArray AndroidUtil::createIntArray(JNIEnv *env, const std::vector<jint> &data) {
|
||||||
|
size_t size = data.size();
|
||||||
|
jintArray array = env->NewIntArray(size);
|
||||||
|
env->SetIntArrayRegion(array, 0, size, &data.front());
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
jbyteArray AndroidUtil::createByteArray(JNIEnv *env, const std::vector<jbyte> &data) {
|
||||||
|
size_t size = data.size();
|
||||||
|
jbyteArray array = env->NewByteArray(size);
|
||||||
|
env->SetByteArrayRegion(array, 0, size, &data.front());
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
jobjectArray AndroidUtil::createStringArray(JNIEnv *env, const std::vector<std::string> &data) {
|
||||||
|
size_t size = data.size();
|
||||||
|
jclass cls = env->FindClass("java/lang/String");
|
||||||
|
jobjectArray array = env->NewObjectArray(size, cls, 0);
|
||||||
|
for (size_t i = 0; i < size; ++i) {
|
||||||
|
const std::string &str = data[i];
|
||||||
|
if (str.length() > 0) {
|
||||||
|
jstring javaStr = env->NewStringUTF(str.c_str());
|
||||||
|
env->SetObjectArrayElement(array, i, javaStr);
|
||||||
|
env->DeleteLocalRef(javaStr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
void AndroidUtil::throwRuntimeException(JNIEnv *env, const std::string &message) {
|
void AndroidUtil::throwRuntimeException(JNIEnv *env, const std::string &message) {
|
||||||
jclass cls = env->FindClass("java/lang/RuntimeException");
|
jclass cls = env->FindClass("java/lang/RuntimeException");
|
||||||
env->ThrowNew(cls, message.c_str());
|
env->ThrowNew(cls, message.c_str());
|
||||||
|
|
|
@ -23,6 +23,7 @@
|
||||||
#include <jni.h>
|
#include <jni.h>
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
class AndroidUtil {
|
class AndroidUtil {
|
||||||
|
|
||||||
|
@ -42,6 +43,7 @@ public:
|
||||||
static const char * const Class_Paths;
|
static const char * const Class_Paths;
|
||||||
static const char * const Class_Book;
|
static const char * const Class_Book;
|
||||||
static const char * const Class_Tag;
|
static const char * const Class_Tag;
|
||||||
|
static const char * const Class_NativeBookModel;
|
||||||
|
|
||||||
static jobject OBJECT_java_lang_System_err;
|
static jobject OBJECT_java_lang_System_err;
|
||||||
|
|
||||||
|
@ -87,6 +89,14 @@ public:
|
||||||
|
|
||||||
static jmethodID SMID_Tag_getTag;
|
static jmethodID SMID_Tag_getTag;
|
||||||
|
|
||||||
|
static jfieldID FID_NativeBookModel_Book;
|
||||||
|
static jmethodID MID_NativeBookModel_initImageMap;
|
||||||
|
static jmethodID MID_NativeBookModel_initInternalHyperlinks;
|
||||||
|
static jmethodID MID_NativeBookModel_initTOC;
|
||||||
|
static jmethodID MID_NativeBookModel_createTextModel;
|
||||||
|
static jmethodID MID_NativeBookModel_setBookTextModel;
|
||||||
|
static jmethodID MID_NativeBookModel_setFootnoteModel;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static bool init(JavaVM* jvm);
|
static bool init(JavaVM* jvm);
|
||||||
static JNIEnv *getEnv();
|
static JNIEnv *getEnv();
|
||||||
|
@ -96,6 +106,10 @@ public:
|
||||||
static jstring createJavaString(JNIEnv* env, const std::string &str);
|
static jstring createJavaString(JNIEnv* env, const std::string &str);
|
||||||
static std::string convertNonUtfString(const std::string &str);
|
static std::string convertNonUtfString(const std::string &str);
|
||||||
|
|
||||||
|
static jintArray createIntArray(JNIEnv *env, const std::vector<jint> &data);
|
||||||
|
static jbyteArray createByteArray(JNIEnv *env, const std::vector<jbyte> &data);
|
||||||
|
static jobjectArray createStringArray(JNIEnv *env, const std::vector<std::string> &data);
|
||||||
|
|
||||||
static void throwRuntimeException(JNIEnv *env, const std::string &message);
|
static void throwRuntimeException(JNIEnv *env, const std::string &message);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
11
proguard.cfg
11
proguard.cfg
|
@ -27,6 +27,7 @@
|
||||||
public ** getPath();
|
public ** getPath();
|
||||||
public long size();
|
public long size();
|
||||||
}
|
}
|
||||||
|
-keep class org.geometerplus.zlibrary.text.model.ZLTextModel
|
||||||
-keep class org.geometerplus.fbreader.formats.PluginCollection
|
-keep class org.geometerplus.fbreader.formats.PluginCollection
|
||||||
-keepclassmembers class org.geometerplus.fbreader.formats.PluginCollection {
|
-keepclassmembers class org.geometerplus.fbreader.formats.PluginCollection {
|
||||||
public static ** Instance();
|
public static ** Instance();
|
||||||
|
@ -55,6 +56,16 @@
|
||||||
-keepclassmembers class org.geometerplus.fbreader.library.Tag {
|
-keepclassmembers class org.geometerplus.fbreader.library.Tag {
|
||||||
public static ** getTag(**,**);
|
public static ** getTag(**,**);
|
||||||
}
|
}
|
||||||
|
-keep class org.geometerplus.fbreader.bookmodel.NativeBookModel
|
||||||
|
-keepclassmembers class org.geometerplus.fbreader.bookmodel.NativeBookModel {
|
||||||
|
public ** Book;
|
||||||
|
public void initImageMap(**[],int[],int[],**,**,int);
|
||||||
|
public void initInternalHyperlinks(**,**,int);
|
||||||
|
public void initTOC(**,int[],int[]);
|
||||||
|
public ** createTextModel(**,**,int,int[],int[],int[],int[],byte[],**,**,int);
|
||||||
|
public void setBookTextModel(**);
|
||||||
|
public void setFootnoteModel(**);
|
||||||
|
}
|
||||||
|
|
||||||
-keepclasseswithmembernames class * {
|
-keepclasseswithmembernames class * {
|
||||||
native <methods>;
|
native <methods>;
|
||||||
|
|
|
@ -19,6 +19,8 @@
|
||||||
|
|
||||||
package org.geometerplus.fbreader.bookmodel;
|
package org.geometerplus.fbreader.bookmodel;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
import org.geometerplus.zlibrary.text.model.*;
|
import org.geometerplus.zlibrary.text.model.*;
|
||||||
|
|
||||||
import org.geometerplus.fbreader.library.Book;
|
import org.geometerplus.fbreader.library.Book;
|
||||||
|
@ -43,6 +45,48 @@ public class NativeBookModel extends BookModelImpl {
|
||||||
myInternalHyperlinks = new CachedCharStorageRO(directoryName, fileExtension, blocksNumber);
|
myInternalHyperlinks = new CachedCharStorageRO(directoryName, fileExtension, blocksNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void initTOC(ZLTextModel contentsModel, int[] childrenNumbers, int[] referenceNumbers) {
|
||||||
|
final StringBuilder buffer = new StringBuilder();
|
||||||
|
|
||||||
|
final ArrayList<Integer> positions = new ArrayList<Integer>();
|
||||||
|
TOCTree tree = TOCTree;
|
||||||
|
|
||||||
|
final int size = contentsModel.getParagraphsNumber();
|
||||||
|
for (int pos = 0; pos < size; ++pos) {
|
||||||
|
positions.add(pos);
|
||||||
|
ZLTextParagraph par = contentsModel.getParagraph(pos);
|
||||||
|
|
||||||
|
buffer.delete(0, buffer.length());
|
||||||
|
ZLTextParagraph.EntryIterator it = par.iterator();
|
||||||
|
while (it.hasNext()) {
|
||||||
|
it.next();
|
||||||
|
if (it.getType() == ZLTextParagraph.Entry.TEXT) {
|
||||||
|
buffer.append(it.getTextData(), it.getTextOffset(), it.getTextLength());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tree = new TOCTree(tree);
|
||||||
|
tree.setText(buffer.toString());
|
||||||
|
tree.setReference(myBookTextModel, referenceNumbers[pos]);
|
||||||
|
|
||||||
|
while (positions.size() > 0 && tree != TOCTree) {
|
||||||
|
final int lastIndex = positions.size() - 1;
|
||||||
|
final int treePos = positions.get(lastIndex);
|
||||||
|
if (tree.subTrees().size() < childrenNumbers[treePos]) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
tree = tree.Parent;
|
||||||
|
positions.remove(lastIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tree != TOCTree || positions.size() > 0) {
|
||||||
|
throw new RuntimeException("Invalid state after TOC building:\n"
|
||||||
|
+ "tree.Level = " + tree.Level + "\n"
|
||||||
|
+ "positions.size() = " + positions.size());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public ZLTextModel createTextModel(
|
public ZLTextModel createTextModel(
|
||||||
String id, String language, int paragraphsNumber,
|
String id, String language, int paragraphsNumber,
|
||||||
int[] entryIndices, int[] entryOffsets,
|
int[] entryIndices, int[] entryOffsets,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue