/* * Copyright (C) 2011-2012 Geometer Plus * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301, USA. */ #include #include #include #include "fbreader/src/bookmodel/BookModel.h" #include "fbreader/src/formats/FormatPlugin.h" #include "fbreader/src/library/Library.h" #include "fbreader/src/library/Author.h" #include "fbreader/src/library/Book.h" #include "fbreader/src/library/Tag.h" static shared_ptr findCppPlugin(JNIEnv *env, jobject base) { jstring fileTypeJava = (jstring)env->CallObjectMethod(base, AndroidUtil::MID_NativeFormatPlugin_supportedFileType); const std::string fileTypeCpp = AndroidUtil::fromJavaString(env, fileTypeJava); env->DeleteLocalRef(fileTypeJava); shared_ptr plugin = PluginCollection::Instance().pluginByType(fileTypeCpp); if (plugin.isNull()) { AndroidUtil::throwRuntimeException(env, "Native FormatPlugin instance is NULL for type " + fileTypeCpp); } return plugin; } static void fillMetaInfo(JNIEnv* env, jobject javaBook, Book &book) { jstring javaString; javaString = AndroidUtil::createJavaString(env, book.title()); env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_setTitle, javaString); env->DeleteLocalRef(javaString); javaString = AndroidUtil::createJavaString(env, book.language()); if (javaString != 0) { env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_setLanguage, javaString); env->DeleteLocalRef(javaString); } javaString = AndroidUtil::createJavaString(env, book.encoding()); if (javaString != 0) { env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_setEncoding, javaString); env->DeleteLocalRef(javaString); } javaString = AndroidUtil::createJavaString(env, book.seriesTitle()); if (javaString != 0) { env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_setSeriesInfo, javaString, (jfloat)book.indexInSeries()); env->DeleteLocalRef(javaString); } const AuthorList &authors = book.authors(); for (size_t i = 0; i < authors.size(); ++i) { const Author &author = *authors[i]; javaString = env->NewStringUTF(author.name().c_str()); jstring key = env->NewStringUTF(author.sortKey().c_str()); env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_addAuthor, javaString, key); env->DeleteLocalRef(key); env->DeleteLocalRef(javaString); } const TagList &tags = book.tags(); for (size_t i = 0; i < tags.size(); ++i) { const Tag &tag = *tags[i]; env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_addTag, tag.javaTag(env)); } } void fillLanguageAndEncoding(JNIEnv* env, jobject javaBook, Book &book) { jstring javaString; javaString = AndroidUtil::createJavaString(env, book.language()); if (javaString != 0) { env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_setLanguage, javaString); env->DeleteLocalRef(javaString); } javaString = AndroidUtil::createJavaString(env, book.encoding()); if (javaString != 0) { env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_setEncoding, javaString); env->DeleteLocalRef(javaString); } env->CallVoidMethod(javaBook, AndroidUtil::MID_Book_save); } extern "C" JNIEXPORT jboolean JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPlugin_readMetaInfoNative(JNIEnv* env, jobject thiz, jobject javaBook) { shared_ptr plugin = findCppPlugin(env, thiz); if (plugin.isNull()) { return JNI_FALSE; } shared_ptr book = Book::loadFromJavaBook(env, javaBook); if (!plugin->readMetaInfo(*book)) { return JNI_FALSE; } fillMetaInfo(env, javaBook, *book); return JNI_TRUE; } extern "C" JNIEXPORT void JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPlugin_detectLanguageAndEncoding(JNIEnv* env, jobject thiz, jobject javaBook) { shared_ptr plugin = findCppPlugin(env, thiz); if (plugin.isNull()) { return; } shared_ptr book = Book::loadFromJavaBook(env, javaBook); if (!plugin->readLanguageAndEncoding(*book)) { return; } fillLanguageAndEncoding(env, javaBook, *book); } 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 &links = model.internalHyperlinks(); std::map::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 childrenNumbers; std::vector 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::createJavaIntArray(env, childrenNumbers); jintArray javaReferenceNumbers = AndroidUtil::createJavaIntArray(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" JNIEXPORT jboolean JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPlugin_readModelNative(JNIEnv* env, jobject thiz, jobject javaModel) { shared_ptr plugin = findCppPlugin(env, thiz); if (plugin.isNull()) { return JNI_FALSE; } jobject javaBook = env->GetObjectField(javaModel, AndroidUtil::FID_NativeBookModel_Book); shared_ptr book = Book::loadFromJavaBook(env, javaBook); shared_ptr model = new BookModel(book, javaModel); if (!plugin->readModel(*model)) { return JNI_FALSE; } model->flush(); if (!initInternalHyperlinks(env, javaModel, *model) || !initTOC(env, javaModel, *model)) { return JNI_FALSE; } shared_ptr 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 > &footnotes = model->footnotes(); std::map >::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" JNIEXPORT void JNICALL Java_org_geometerplus_fbreader_formats_NativeFormatPlugin_readCoverInternal(JNIEnv* env, jobject thiz, jobject file, jobjectArray box) { shared_ptr plugin = findCppPlugin(env, thiz); if (plugin.isNull()) { return; } jstring javaPath = (jstring)env->CallObjectMethod(file, AndroidUtil::MID_ZLFile_getPath); const std::string path = AndroidUtil::fromJavaString(env, javaPath); env->DeleteLocalRef(javaPath); shared_ptr image = plugin->coverImage(ZLFile(path)); if (!image.isNull()) { jobject javaImage = AndroidUtil::createJavaImage(env, (const ZLFileImage&)*image); env->SetObjectArrayElement(box, 0, javaImage); env->DeleteLocalRef(javaImage); } }