/* * 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 "AndroidUtil.h" #include "JniEnvelope.h" JavaVM *AndroidUtil::ourJavaVM = 0; shared_ptr AndroidUtil::Class_java_lang_String; const char * const AndroidUtil::Class_java_util_Collection = "java/util/Collection"; const char * const AndroidUtil::Class_java_util_Locale = "java/util/Locale"; const char * const AndroidUtil::Class_java_io_InputStream = "java/io/InputStream"; const char * const AndroidUtil::Class_ZLibrary = "org/geometerplus/zlibrary/core/library/ZLibrary"; shared_ptr AndroidUtil::Class_NativeFormatPlugin; const char * const AndroidUtil::Class_PluginCollection = "org/geometerplus/fbreader/formats/PluginCollection"; const char * const AndroidUtil::Class_Encoding = "org/geometerplus/zlibrary/core/encodings/Encoding"; const char * const AndroidUtil::Class_EncodingConverter = "org/geometerplus/zlibrary/core/encodings/EncodingConverter"; const char * const AndroidUtil::Class_JavaEncodingCollection = "org/geometerplus/zlibrary/core/encodings/JavaEncodingCollection"; const char * const AndroidUtil::Class_Paths = "org/geometerplus/fbreader/Paths"; const char * const AndroidUtil::Class_ZLFile = "org/geometerplus/zlibrary/core/filesystem/ZLFile"; shared_ptr AndroidUtil::Class_ZLFileImage; 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_NativeBookModel = "org/geometerplus/fbreader/bookmodel/NativeBookModel"; //const char * const AndroidUtil::Class_BookReadingException = "org/geometerplus/fbreader/bookmodel/BookReadingException"; shared_ptr AndroidUtil::Method_java_lang_String_toLowerCase; shared_ptr AndroidUtil::Method_java_lang_String_toUpperCase; jmethodID AndroidUtil::MID_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::Constructor_NativeFormatPlugin; shared_ptr AndroidUtil::Method_NativeFormatPlugin_supportedFileType; shared_ptr AndroidUtil::StaticMethod_PluginCollection_Instance; shared_ptr AndroidUtil::Method_Encoding_createConverter; jfieldID AndroidUtil::FID_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_String; 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; jfieldID AndroidUtil::FID_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::Method_Book_save; shared_ptr AndroidUtil::StaticMethod_Tag_getTag; jfieldID AndroidUtil::FID_NativeBookModel_Book; shared_ptr AndroidUtil::Method_NativeBookModel_initInternalHyperlinks; shared_ptr AndroidUtil::Method_NativeBookModel_initTOC; 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; } #define CHECK_NULL(value) if ((value) == 0) { return false; } bool AndroidUtil::init(JavaVM* jvm) { ourJavaVM = jvm; JNIEnv *env = getEnv(); jclass cls; Class_java_lang_String = new JavaClass(env, "java/lang/String"); Method_java_lang_String_toLowerCase = new StringMethod(*Class_java_lang_String, "toLowerCase", "()"); Method_java_lang_String_toUpperCase = new StringMethod(*Class_java_lang_String, "toUpperCase", "()"); CHECK_NULL( cls = env->FindClass(Class_java_util_Collection) ); CHECK_NULL( MID_java_util_Collection_toArray = env->GetMethodID(cls, "toArray", "()[Ljava/lang/Object;") ); //CHECK_NULL( MID_java_util_Collection_add = env->GetMethodID(cls, "add", "(Ljava/lang/Object;)Z") ); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_java_util_Locale) ); StaticMethod_java_util_Locale_getDefault = new StaticObjectMethod(env, cls, "getDefault", "java/util/Locale", "()"); Method_java_util_Locale_getLanguage = new StringMethod(env, cls, "getLanguage", "()"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_java_io_InputStream) ); Method_java_io_InputStream_close = new VoidMethod(env, cls, "close", "()"); Method_java_io_InputStream_read = new IntMethod(env, cls, "read", "([BII)"); Method_java_io_InputStream_skip = new LongMethod(env, cls, "skip", "(J)"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_ZLibrary) ); StaticMethod_ZLibrary_Instance = new StaticObjectMethod(env, cls, "Instance", "org/geometerplus/zlibrary/core/library/ZLibrary", "()"); Method_ZLibrary_getVersionName = new StringMethod(env, cls, "getVersionName", "()"); env->DeleteLocalRef(cls); Class_NativeFormatPlugin = new JavaClass(env, "org/geometerplus/fbreader/formats/NativeFormatPlugin"); Constructor_NativeFormatPlugin = new Constructor(*Class_NativeFormatPlugin, "(Ljava/lang/String;)V"); Method_NativeFormatPlugin_supportedFileType = new StringMethod(*Class_NativeFormatPlugin, "supportedFileType", "()"); CHECK_NULL( cls = env->FindClass(Class_PluginCollection) ); StaticMethod_PluginCollection_Instance = new StaticObjectMethod(env, cls, "Instance", "org/geometerplus/fbreader/formats/PluginCollection", "()"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_Encoding) ); Method_Encoding_createConverter = new ObjectMethod(env, cls, "createConverter", "org/geometerplus/zlibrary/core/encodings/EncodingConverter", "()"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_EncodingConverter) ); CHECK_NULL( FID_EncodingConverter_Name = env->GetFieldID(cls, "Name", "Ljava/lang/String;") ); Method_EncodingConverter_convert = new IntMethod(env, cls, "convert", "([BII[BI)"); Method_EncodingConverter_reset = new VoidMethod(env, cls, "reset", "()"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_JavaEncodingCollection) ); StaticMethod_JavaEncodingCollection_Instance = new StaticObjectMethod(env, cls, "Instance", "org/geometerplus/zlibrary/core/encodings/JavaEncodingCollection", "()"); Method_JavaEncodingCollection_getEncoding_String = new ObjectMethod(env, cls, "getEncoding", "org/geometerplus/zlibrary/core/encodings/Encoding", "(Ljava/lang/String;)"); Method_JavaEncodingCollection_getEncoding_int = new ObjectMethod(env, cls, "getEncoding", "org/geometerplus/zlibrary/core/encodings/Encoding", "(I)"); Method_JavaEncodingCollection_providesConverterFor = new BooleanMethod(env, cls, "providesConverterFor", "(Ljava/lang/String;)"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_ZLFile) ); StaticMethod_ZLFile_createFileByPath = new StaticObjectMethod(env, cls, "createFileByPath", "org/geometerplus/zlibrary/core/filesystem/ZLFile", "(Ljava/lang/String;)"); Method_ZLFile_children = new ObjectMethod(env, cls, "children", "java/util/List", "()"); Method_ZLFile_exists = new BooleanMethod(env, cls, "exists", "()"); Method_ZLFile_isDirectory = new BooleanMethod(env, cls, "isDirectory", "()"); Method_ZLFile_getInputStream = new ObjectMethod(env, cls, "getInputStream", "java/io/InputStream", "()"); Method_ZLFile_getPath = new StringMethod(env, cls, "getPath", "()"); Method_ZLFile_size = new LongMethod(env, cls, "size", "()"); env->DeleteLocalRef(cls); Class_ZLFileImage = new JavaClass(env, "org/geometerplus/zlibrary/core/image/ZLFileImage"); Constructor_ZLFileImage = new Constructor(*Class_ZLFileImage, "(Ljava/lang/String;Lorg/geometerplus/zlibrary/core/filesystem/ZLFile;Ljava/lang/String;II)V"); CHECK_NULL( cls = env->FindClass(Class_Paths) ); StaticMethod_Paths_cacheDirectory = new StaticObjectMethod(env, cls, "cacheDirectory", "java/lang/String", "()"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_Book) ); CHECK_NULL( FID_Book_File = env->GetFieldID(cls, "File", "Lorg/geometerplus/zlibrary/core/filesystem/ZLFile;") ); Method_Book_getTitle = new StringMethod(env, cls, "getTitle", "()"); Method_Book_getLanguage = new StringMethod(env, cls, "getLanguage", "()"); Method_Book_getEncodingNoDetection = new StringMethod(env, cls, "getEncodingNoDetection", "()"); Method_Book_setTitle = new VoidMethod(env, cls, "setTitle", "(Ljava/lang/String;)"); Method_Book_setSeriesInfo = new VoidMethod(env, cls, "setSeriesInfo", "(Ljava/lang/String;F)"); Method_Book_setLanguage = new VoidMethod(env, cls, "setLanguage", "(Ljava/lang/String;)"); Method_Book_setEncoding = new VoidMethod(env, cls, "setEncoding", "(Ljava/lang/String;)"); Method_Book_addAuthor = new VoidMethod(env, cls, "addAuthor", "(Ljava/lang/String;Ljava/lang/String;)"); Method_Book_addTag = new VoidMethod(env, cls, "addTag", "(Lorg/geometerplus/fbreader/library/Tag;)"); Method_Book_save = new BooleanMethod(env, cls, "save", "()"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_Tag) ); StaticMethod_Tag_getTag = new StaticObjectMethod(env, cls, "getTag", "org/geometerplus/fbreader/library/Tag", "(Lorg/geometerplus/fbreader/library/Tag;Ljava/lang/String;)"); env->DeleteLocalRef(cls); CHECK_NULL( cls = env->FindClass(Class_NativeBookModel) ); CHECK_NULL( FID_NativeBookModel_Book = env->GetFieldID(cls, "Book", "Lorg/geometerplus/fbreader/library/Book;") ); Method_NativeBookModel_initInternalHyperlinks = new VoidMethod(env, cls, "initInternalHyperlinks", "(Ljava/lang/String;Ljava/lang/String;I)"); Method_NativeBookModel_initTOC = new VoidMethod(env, cls, "initTOC", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;[I[I)"); Method_NativeBookModel_createTextModel = new ObjectMethod(env, cls, "createTextModel", "org/geometerplus/zlibrary/text/model/ZLTextModel", "(Ljava/lang/String;Ljava/lang/String;I[I[I[I[I[BLjava/lang/String;Ljava/lang/String;I)"); Method_NativeBookModel_setBookTextModel = new VoidMethod(env, cls, "setBookTextModel", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;)"); Method_NativeBookModel_setFootnoteModel = new VoidMethod(env, cls, "setFootnoteModel", "(Lorg/geometerplus/zlibrary/text/model/ZLTextModel;)"); Method_NativeBookModel_addImage = new VoidMethod(env, cls, "addImage", "(Ljava/lang/String;Lorg/geometerplus/zlibrary/core/image/ZLImage;)"); env->DeleteLocalRef(cls); /* CHECK_NULL( cls = env->FindClass(Class_BookReadingException) ); CHECK_NULL( StaticMethod_BookReadingException_throwForFile = env->GetStaticMethodID(cls, "throwForFile", "(Ljava/lang/String;Lorg/geometerplus/zlibrary/core/filesystem/ZLFile;)V") ); env->DeleteLocalRef(cls); */ return true; } jobject AndroidUtil::createJavaFile(JNIEnv *env, const std::string &path) { jstring javaPath = env->NewStringUTF(path.c_str()); jclass cls = env->FindClass(Class_ZLFile); jobject javaFile = StaticMethod_ZLFile_createFileByPath->call(cls, javaPath); env->DeleteLocalRef(cls); 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()); jobject javaImage = Constructor_ZLFileImage->call( javaMimeType, javaFile, javaEncoding, image.offset(), image.size() ); env->DeleteLocalRef(javaEncoding); env->DeleteLocalRef(javaFile); env->DeleteLocalRef(javaMimeType); 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] = 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) { 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) { size_t size = data.size(); jbyteArray array = env->NewByteArray(size); env->SetByteArrayRegion(array, 0, size, &data.front()); return array; } void AndroidUtil::throwRuntimeException(JNIEnv *env, const std::string &message) { // TODO: possible memory leak jclass cls = env->FindClass("java/lang/RuntimeException"); env->ThrowNew(cls, message.c_str()); } /* void AndroidUtil::throwBookReadingException(const std::string &resourceId, const ZLFile &file) { JNIEnv *env = getEnv(); jclass cls = env->FindClass("org/geometerplus/fbreader/bookmodel/BookReadingException"); env->CallStaticVoidMethod( cls, StaticMethod_BookReadingException_throwForFile, AndroidUtil::createJavaString(env, resourceId), AndroidUtil::createJavaFile(env, file.path()) ); // TODO: possible memory leak // TODO: clear cls & ZLFile object references } */