/* * Copyright (C) 2011-2013 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 "AndroidUtil.h" #include "JniEnvelope.h" JavaVM *AndroidUtil::ourJavaVM = 0; JavaClass AndroidUtil::Class_java_lang_Object("java/lang/Object"); JavaArray AndroidUtil::Array_java_lang_Object(Class_java_lang_Object); JavaClass AndroidUtil::Class_java_lang_RuntimeException("java/lang/RuntimeException"); JavaClass AndroidUtil::Class_java_lang_String("java/lang/String"); JavaClass AndroidUtil::Class_java_util_Collection("java/util/Collection"); JavaClass AndroidUtil::Class_java_util_List("java/util/List"); JavaClass AndroidUtil::Class_java_util_Locale("java/util/Locale"); JavaClass AndroidUtil::Class_java_io_InputStream("java/io/InputStream"); JavaClass AndroidUtil::Class_ZLibrary("org/geometerplus/zlibrary/core/library/ZLibrary"); JavaClass AndroidUtil::Class_ZLFile("org/geometerplus/zlibrary/core/filesystem/ZLFile"); JavaClass AndroidUtil::Class_ZLFileImage("org/geometerplus/zlibrary/core/image/ZLFileImage"); JavaClass AndroidUtil::Class_ZLTextModel("org/geometerplus/zlibrary/text/model/ZLTextModel"); JavaClass AndroidUtil::Class_CachedCharStorageException("org/geometerplus/zlibrary/text/model/CachedCharStorageException"); JavaClass AndroidUtil::Class_Encoding("org/geometerplus/zlibrary/core/encodings/Encoding"); JavaClass AndroidUtil::Class_EncodingConverter("org/geometerplus/zlibrary/core/encodings/EncodingConverter"); JavaClass AndroidUtil::Class_JavaEncodingCollection("org/geometerplus/zlibrary/core/encodings/JavaEncodingCollection"); JavaClass AndroidUtil::Class_NativeFormatPlugin("org/geometerplus/fbreader/formats/NativeFormatPlugin"); JavaClass AndroidUtil::Class_PluginCollection("org/geometerplus/fbreader/formats/PluginCollection"); JavaClass AndroidUtil::Class_Paths("org/geometerplus/fbreader/Paths"); JavaClass AndroidUtil::Class_Book("org/geometerplus/fbreader/book/Book"); JavaClass AndroidUtil::Class_Tag("org/geometerplus/fbreader/book/Tag"); JavaClass AndroidUtil::Class_NativeBookModel("org/geometerplus/fbreader/bookmodel/NativeBookModel"); shared_ptr AndroidUtil::Method_java_lang_String_toLowerCase; shared_ptr AndroidUtil::Method_java_lang_String_toUpperCase; shared_ptr AndroidUtil::Method_java_util_Collection_toArray; shared_ptr AndroidUtil::StaticMethod_java_util_Locale_getDefault; shared_ptr AndroidUtil::Method_java_util_Locale_getLanguage; shared_ptr AndroidUtil::Method_java_io_InputStream_close; shared_ptr AndroidUtil::Method_java_io_InputStream_read; shared_ptr AndroidUtil::Method_java_io_InputStream_skip; shared_ptr AndroidUtil::StaticMethod_ZLibrary_Instance; shared_ptr AndroidUtil::Method_ZLibrary_getVersionName; shared_ptr AndroidUtil::StaticMethod_NativeFormatPlugin_create; shared_ptr AndroidUtil::Method_NativeFormatPlugin_supportedFileType; shared_ptr AndroidUtil::StaticMethod_PluginCollection_Instance; shared_ptr AndroidUtil::Method_Encoding_createConverter; shared_ptr AndroidUtil::Field_EncodingConverter_Name; shared_ptr AndroidUtil::Method_EncodingConverter_convert; shared_ptr AndroidUtil::Method_EncodingConverter_reset; shared_ptr AndroidUtil::StaticMethod_JavaEncodingCollection_Instance; //shared_ptr AndroidUtil::Method_JavaEncodingCollection_getEncoding_int; shared_ptr AndroidUtil::Method_JavaEncodingCollection_getEncoding; shared_ptr AndroidUtil::Method_JavaEncodingCollection_providesConverterFor; shared_ptr AndroidUtil::StaticMethod_ZLFile_createFileByPath; shared_ptr AndroidUtil::Method_ZLFile_children; shared_ptr AndroidUtil::Method_ZLFile_exists; shared_ptr AndroidUtil::Method_ZLFile_getInputStream; shared_ptr AndroidUtil::Method_ZLFile_getPath; shared_ptr AndroidUtil::Method_ZLFile_isDirectory; shared_ptr AndroidUtil::Method_ZLFile_size; shared_ptr AndroidUtil::Constructor_ZLFileImage; shared_ptr AndroidUtil::StaticMethod_Paths_cacheDirectory; shared_ptr AndroidUtil::Field_Book_File; shared_ptr AndroidUtil::Method_Book_getTitle; shared_ptr AndroidUtil::Method_Book_getLanguage; shared_ptr AndroidUtil::Method_Book_getEncodingNoDetection; shared_ptr AndroidUtil::Method_Book_setTitle; shared_ptr AndroidUtil::Method_Book_setSeriesInfo; shared_ptr AndroidUtil::Method_Book_setLanguage; shared_ptr AndroidUtil::Method_Book_setEncoding; shared_ptr AndroidUtil::Method_Book_addAuthor; shared_ptr AndroidUtil::Method_Book_addTag; shared_ptr AndroidUtil::StaticMethod_Tag_getTag; shared_ptr AndroidUtil::Field_NativeBookModel_Book; shared_ptr AndroidUtil::Method_NativeBookModel_initInternalHyperlinks; shared_ptr AndroidUtil::Method_NativeBookModel_addTOCItem; shared_ptr AndroidUtil::Method_NativeBookModel_leaveTOCItem; shared_ptr AndroidUtil::Method_NativeBookModel_createTextModel; shared_ptr AndroidUtil::Method_NativeBookModel_setBookTextModel; shared_ptr AndroidUtil::Method_NativeBookModel_setFootnoteModel; shared_ptr AndroidUtil::Method_NativeBookModel_addImage; //shared_ptr AndroidUtil::StaticMethod_BookReadingException_throwForFile; JNIEnv *AndroidUtil::getEnv() { JNIEnv *env; ourJavaVM->GetEnv((void **)&env, JNI_VERSION_1_2); return env; } bool AndroidUtil::init(JavaVM* jvm) { ourJavaVM = jvm; Method_java_lang_String_toLowerCase = new StringMethod(Class_java_lang_String, "toLowerCase", "()"); Method_java_lang_String_toUpperCase = new StringMethod(Class_java_lang_String, "toUpperCase", "()"); Method_java_util_Collection_toArray = new ObjectArrayMethod(Class_java_util_Collection, "toArray", Array_java_lang_Object, "()"); StaticMethod_java_util_Locale_getDefault = new StaticObjectMethod(Class_java_util_Locale, "getDefault", Class_java_util_Locale, "()"); Method_java_util_Locale_getLanguage = new StringMethod(Class_java_util_Locale, "getLanguage", "()"); Method_java_io_InputStream_close = new VoidMethod(Class_java_io_InputStream, "close", "()"); Method_java_io_InputStream_read = new IntMethod(Class_java_io_InputStream, "read", "([BII)"); Method_java_io_InputStream_skip = new LongMethod(Class_java_io_InputStream, "skip", "(J)"); StaticMethod_ZLibrary_Instance = new StaticObjectMethod(Class_ZLibrary, "Instance", Class_ZLibrary, "()"); Method_ZLibrary_getVersionName = new StringMethod(Class_ZLibrary, "getVersionName", "()"); StaticMethod_NativeFormatPlugin_create = new StaticObjectMethod(Class_NativeFormatPlugin, "create", Class_NativeFormatPlugin, "(Ljava/lang/String;)"); Method_NativeFormatPlugin_supportedFileType = new StringMethod(Class_NativeFormatPlugin, "supportedFileType", "()"); StaticMethod_PluginCollection_Instance = new StaticObjectMethod(Class_PluginCollection, "Instance", Class_PluginCollection, "()"); Method_Encoding_createConverter = new ObjectMethod(Class_Encoding, "createConverter", Class_EncodingConverter, "()"); Field_EncodingConverter_Name = new ObjectField(Class_EncodingConverter, "Name", Class_java_lang_String); Method_EncodingConverter_convert = new IntMethod(Class_EncodingConverter, "convert", "([BII[C)"); Method_EncodingConverter_reset = new VoidMethod(Class_EncodingConverter, "reset", "()"); StaticMethod_JavaEncodingCollection_Instance = new StaticObjectMethod(Class_JavaEncodingCollection, "Instance", Class_JavaEncodingCollection, "()"); Method_JavaEncodingCollection_getEncoding = new ObjectMethod(Class_JavaEncodingCollection, "getEncoding", Class_Encoding, "(Ljava/lang/String;)"); //Method_JavaEncodingCollection_getEncoding_int = new ObjectMethod(Class_JavaEncodingCollection, "getEncoding", Class_Encoding, "(I)"); Method_JavaEncodingCollection_providesConverterFor = new BooleanMethod(Class_JavaEncodingCollection, "providesConverterFor", "(Ljava/lang/String;)"); StaticMethod_ZLFile_createFileByPath = new StaticObjectMethod(Class_ZLFile, "createFileByPath", Class_ZLFile, "(Ljava/lang/String;)"); Method_ZLFile_children = new ObjectMethod(Class_ZLFile, "children", Class_java_util_List, "()"); Method_ZLFile_exists = new BooleanMethod(Class_ZLFile, "exists", "()"); Method_ZLFile_isDirectory = new BooleanMethod(Class_ZLFile, "isDirectory", "()"); Method_ZLFile_getInputStream = new ObjectMethod(Class_ZLFile, "getInputStream", Class_java_io_InputStream, "()"); Method_ZLFile_getPath = new StringMethod(Class_ZLFile, "getPath", "()"); Method_ZLFile_size = new LongMethod(Class_ZLFile, "size", "()"); Constructor_ZLFileImage = new Constructor(Class_ZLFileImage, "(Ljava/lang/String;Lorg/geometerplus/zlibrary/core/filesystem/ZLFile;Ljava/lang/String;[I[I)V"); StaticMethod_Paths_cacheDirectory = new StaticObjectMethod(Class_Paths, "cacheDirectory", Class_java_lang_String, "()"); Field_Book_File = new ObjectField(Class_Book, "File", Class_ZLFile); Method_Book_getTitle = new StringMethod(Class_Book, "getTitle", "()"); Method_Book_getLanguage = new StringMethod(Class_Book, "getLanguage", "()"); Method_Book_getEncodingNoDetection = new StringMethod(Class_Book, "getEncodingNoDetection", "()"); Method_Book_setTitle = new VoidMethod(Class_Book, "setTitle", "(Ljava/lang/String;)"); Method_Book_setSeriesInfo = new VoidMethod(Class_Book, "setSeriesInfo", "(Ljava/lang/String;Ljava/lang/String;)"); Method_Book_setLanguage = new VoidMethod(Class_Book, "setLanguage", "(Ljava/lang/String;)"); Method_Book_setEncoding = new VoidMethod(Class_Book, "setEncoding", "(Ljava/lang/String;)"); Method_Book_addAuthor = new VoidMethod(Class_Book, "addAuthor", "(Ljava/lang/String;Ljava/lang/String;)"); Method_Book_addTag = new VoidMethod(Class_Book, "addTag", "(Lorg/geometerplus/fbreader/book/Tag;)"); StaticMethod_Tag_getTag = new StaticObjectMethod(Class_Tag, "getTag", Class_Tag, "(Lorg/geometerplus/fbreader/book/Tag;Ljava/lang/String;)"); Field_NativeBookModel_Book = new ObjectField(Class_NativeBookModel, "Book", Class_Book); Method_NativeBookModel_initInternalHyperlinks = new VoidMethod(Class_NativeBookModel, "initInternalHyperlinks", "(Ljava/lang/String;Ljava/lang/String;I)"); Method_NativeBookModel_addTOCItem = new VoidMethod(Class_NativeBookModel, "addTOCItem", "(Ljava/lang/String;I)"); Method_NativeBookModel_leaveTOCItem = new VoidMethod(Class_NativeBookModel, "leaveTOCItem", "()"); Method_NativeBookModel_createTextModel = new ObjectMethod(Class_NativeBookModel, "createTextModel", Class_ZLTextModel, "(Ljava/lang/String;Ljava/lang/String;I[I[I[I[I[BLjava/lang/String;Ljava/lang/String;I)"); Method_NativeBookModel_setBookTextModel = new VoidMethod(Class_NativeBookModel, "setBookTextModel", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;)"); Method_NativeBookModel_setFootnoteModel = new VoidMethod(Class_NativeBookModel, "setFootnoteModel", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;)"); Method_NativeBookModel_addImage = new VoidMethod(Class_NativeBookModel, "addImage", "(Ljava/lang/String;Lorg/geometerplus/zlibrary/core/image/ZLImage;)"); /* Class_BookReadingException = new JavaClass(env, "org/geometerplus/fbreader/bookmodel/BookReadingException"); StaticMethod_BookReadingException_throwForFile = new StaticVoidMethod(Class_BookReadingException, "throwForFile", "(Ljava/lang/String;Lorg/geometerplus/zlibrary/core/filesystem/ZLFile;)V") ); */ return true; } jobject AndroidUtil::createJavaFile(JNIEnv *env, const std::string &path) { jstring javaPath = createJavaString(env, path); jobject javaFile = StaticMethod_ZLFile_createFileByPath->call(javaPath); env->DeleteLocalRef(javaPath); return javaFile; } jobject AndroidUtil::createJavaImage(JNIEnv *env, const ZLFileImage &image) { jstring javaMimeType = createJavaString(env, image.mimeType()); jobject javaFile = createJavaFile(env, image.file().path()); jstring javaEncoding = createJavaString(env, image.encoding()); std::vector offsets, sizes; const ZLFileImage::Blocks &blocks = image.blocks(); for (std::size_t i = 0; i < blocks.size(); ++i) { offsets.push_back((jint)blocks.at(i).offset); sizes.push_back((jint)blocks.at(i).size); } jintArray javaOffsets = createJavaIntArray(env, offsets); jintArray javaSizes = createJavaIntArray(env, sizes); jobject javaImage = Constructor_ZLFileImage->call( javaMimeType, javaFile, javaEncoding, javaOffsets, javaSizes ); env->DeleteLocalRef(javaEncoding); env->DeleteLocalRef(javaFile); env->DeleteLocalRef(javaMimeType); env->DeleteLocalRef(javaOffsets); env->DeleteLocalRef(javaSizes); return javaImage; } std::string AndroidUtil::fromJavaString(JNIEnv *env, jstring from) { if (from == 0) { return std::string(); } const char *data = env->GetStringUTFChars(from, 0); const std::string result(data); env->ReleaseStringUTFChars(from, data); return result; } jstring AndroidUtil::createJavaString(JNIEnv* env, const std::string &str) { if (str.empty()) { return 0; } return env->NewStringUTF(str.c_str()); } std::string AndroidUtil::convertNonUtfString(const std::string &str) { const int len = str.length(); if (len == 0) { return str; } JNIEnv *env = getEnv(); jchar *chars = new jchar[len]; for (int i = 0; i < len; ++i) { chars[i] = (unsigned char)str[i]; } jstring javaString = env->NewString(chars, len); const std::string result = fromJavaString(env, javaString); env->DeleteLocalRef(javaString); delete[] chars; return result; } jintArray AndroidUtil::createJavaIntArray(JNIEnv *env, const std::vector &data) { std::size_t size = data.size(); jintArray array = env->NewIntArray(size); env->SetIntArrayRegion(array, 0, size, &data.front()); return array; } jbyteArray AndroidUtil::createJavaByteArray(JNIEnv *env, const std::vector &data) { std::size_t size = data.size(); jbyteArray array = env->NewByteArray(size); env->SetByteArrayRegion(array, 0, size, &data.front()); return array; } void AndroidUtil::throwRuntimeException(const std::string &message) { getEnv()->ThrowNew(Class_java_lang_RuntimeException.j(), message.c_str()); } void AndroidUtil::throwCachedCharStorageException(const std::string &message) { getEnv()->ThrowNew(Class_CachedCharStorageException.j(), message.c_str()); } /* void AndroidUtil::throwBookReadingException(const std::string &resourceId, const ZLFile &file) { JNIEnv *env = getEnv(); env->CallStaticVoidMethod( StaticMethod_BookReadingException_throwForFile, AndroidUtil::createJavaString(env, resourceId), AndroidUtil::createJavaFile(env, file.path()) ); // TODO: possible memory leak // TODO: clear ZLFile object reference } */