diff --git a/android/AndroidManifest.xml b/android/AndroidManifest.xml index 141ff315fc..1d38f3aef5 100644 --- a/android/AndroidManifest.xml +++ b/android/AndroidManifest.xml @@ -13,6 +13,7 @@ + GetObjectClass(obj); + GetCurrentThreadJNIEnv()->GetMethodID(k, m_name, m_signature); + CHECK(m_index, ("Error: No valid function pointer in ", m_name)); + } + bool Method::CallBoolean(jobject self) { JNIEnv* jniEnv = GetCurrentThreadJNIEnv(); @@ -39,4 +50,9 @@ namespace jni return (int)jniEnv->CallIntMethod(self, m_index); } + + jmethodID Method::GetMethodID() const + { + return m_index; + } } diff --git a/android/jni/com/mapswithme/jni/jni_method.hpp b/android/jni/com/mapswithme/jni/jni_method.hpp index 80c83e60f5..dfe2701c90 100644 --- a/android/jni/com/mapswithme/jni/jni_method.hpp +++ b/android/jni/com/mapswithme/jni/jni_method.hpp @@ -23,8 +23,12 @@ namespace jni public: Method(jclass klass, - const char* name, - const char* signature); + char const * name, + char const * signature); + + Method(jobject obj, + char const * name, + char const * signature); void CallVoid(jobject self) { @@ -61,6 +65,8 @@ namespace jni GetCurrentThreadJNIEnv()->CallVoidMethod(self, m_index, a1, a2, a3, a4, a5); } + jmethodID GetMethodID() const; + bool CallBoolean(jobject self); bool CallInt(jobject self); }; diff --git a/android/jni/com/mapswithme/jni/jni_thread.cpp b/android/jni/com/mapswithme/jni/jni_thread.cpp index bc6b8d2e0b..6e03df502f 100644 --- a/android/jni/com/mapswithme/jni/jni_thread.cpp +++ b/android/jni/com/mapswithme/jni/jni_thread.cpp @@ -20,6 +20,8 @@ namespace jni JavaVM * GetCurrentJVM() { + if (s_jvm == 0) + LOG(LINFO, ("no current JVM")); return s_jvm; } diff --git a/android/jni/com/mapswithme/maps/DownloadUI.cpp b/android/jni/com/mapswithme/maps/DownloadUI.cpp index 97b2d9a564..4da3129cf4 100644 --- a/android/jni/com/mapswithme/maps/DownloadUI.cpp +++ b/android/jni/com/mapswithme/maps/DownloadUI.cpp @@ -7,6 +7,58 @@ #include #include "Framework.hpp" +#include "DownloadUI.hpp" +#include "../jni/jni_thread.hpp" +#include "../../../../../std/bind.hpp" +#include "../../../../../base/logging.hpp" + +android::DownloadUI * g_downloadUI = 0; + +namespace android +{ + DownloadUI::DownloadUI(jobject self) + { + m_self = jni::GetCurrentThreadJNIEnv()->NewGlobalRef(self); + + jclass k = jni::GetCurrentThreadJNIEnv()->GetObjectClass(m_self); + + m_onChangeCountry.reset(new jni::Method(k, "onChangeCountry", "(III)V")); + m_onProgress.reset(new jni::Method(k, "onProgress", "(IIIJJ)V")); + + ASSERT(!g_downloadUI, ()); + g_downloadUI = this; + } + + DownloadUI::~DownloadUI() + { + jni::GetCurrentThreadJNIEnv()->DeleteGlobalRef(m_self); + g_downloadUI = 0; + } + + void DownloadUI::OnChangeCountry(storage::TIndex const & idx) + { + jint group = idx.m_group; + jint country = idx.m_country; + jint region = idx.m_region; + + LOG(LINFO, ("Changed Country", group, country, region)); + + m_onChangeCountry->CallVoid(m_self, group, country, region); + } + + void DownloadUI::OnProgress(storage::TIndex const & idx, pair const & p) + { + jint group = idx.m_group; + jint country = idx.m_country; + jint region = idx.m_region; + jlong p1 = p.first; + jlong p2 = p.second; + + LOG(LINFO, ("Country Progress", group, country, region, p1, p2)); + + m_onProgress->CallVoid(m_self, group, country, region, p1, p2); + } +} /////////////////////////////////////////////////////////////////////////////////// // DownloadUI @@ -14,6 +66,21 @@ extern "C" { + JNIEXPORT void JNICALL + Java_com_mapswithme_maps_DownloadUI_nativeCreate(JNIEnv * env, jobject thiz) + { + g_downloadUI = new android::DownloadUI(thiz); + g_framework->Storage().Subscribe(bind(&android::DownloadUI::OnChangeCountry, g_downloadUI, _1), + bind(&android::DownloadUI::OnProgress, g_downloadUI, _1, _2)); + } + + JNIEXPORT void JNICALL + Java_com_mapswithme_maps_DownloadUI_nativeDestroy(JNIEnv * env, jobject thiz) + { + g_framework->Storage().Unsubscribe(); + delete g_downloadUI; + } + JNIEXPORT jint JNICALL Java_com_mapswithme_maps_DownloadUI_countriesCount(JNIEnv * env, jobject thiz, jint group, jint country, jint region) @@ -30,14 +97,19 @@ extern "C" } JNIEXPORT jlong JNICALL - Java_com_mapswithme_maps_DownloadUI_countrySizeInBytes(JNIEnv * env, jobject thiz, + Java_com_mapswithme_maps_DownloadUI_countryLocalSizeInBytes(JNIEnv * env, jobject thiz, jint group, jint country, jint region) { storage::LocalAndRemoteSizeT const s = g_framework->Storage().CountrySizeInBytes(storage::TIndex(group, country, region)); - // lower int contains remote size, and upper - local one - int64_t mergedSize = s.second; - mergedSize |= (s.first << 32); - return mergedSize; + return s.first; + } + + JNIEXPORT jlong JNICALL + Java_com_mapswithme_maps_DownloadUI_countryRemoteSizeInBytes(JNIEnv * env, jobject thiz, + jint group, jint country, jint region) + { + storage::LocalAndRemoteSizeT const s = g_framework->Storage().CountrySizeInBytes(storage::TIndex(group, country, region)); + return s.second; } JNIEXPORT jint JNICALL @@ -46,5 +118,12 @@ extern "C" { return static_cast(g_framework->Storage().CountryStatus(storage::TIndex(group, country, region))); } + + JNIEXPORT jint JNICALL + Java_com_mapswithme_maps_DownloadUI_downloadCountry(JNIEnv * env, jobject thiz, + jint group, jint country, jint region) + { + g_framework->Storage().DownloadCountry(storage::TIndex(group, country, region)); + } } diff --git a/android/jni/com/mapswithme/maps/DownloadUI.hpp b/android/jni/com/mapswithme/maps/DownloadUI.hpp new file mode 100644 index 0000000000..3a6aebabbe --- /dev/null +++ b/android/jni/com/mapswithme/maps/DownloadUI.hpp @@ -0,0 +1,28 @@ +#pragma once + +#include "Framework.hpp" +#include "../jni/jni_method.hpp" +#include "../../../../../base/base.hpp" +#include "../../../../../std/scoped_ptr.hpp" + +namespace android +{ + class DownloadUI + { + private: + jobject m_self; + + scoped_ptr m_onChangeCountry; + scoped_ptr m_onProgress; + + public: + + DownloadUI(jobject self); + ~DownloadUI(); + + void OnChangeCountry(storage::TIndex const & idx); + void OnProgress(storage::TIndex const & idx, pair const & p); + }; +} + +extern android::DownloadUI * g_downloadUI; diff --git a/android/jni/com/mapswithme/platform/http_thread_android.cpp b/android/jni/com/mapswithme/platform/http_thread_android.cpp index 05205c8dd4..e47a1d6ce3 100644 --- a/android/jni/com/mapswithme/platform/http_thread_android.cpp +++ b/android/jni/com/mapswithme/platform/http_thread_android.cpp @@ -3,23 +3,101 @@ #include "../../../../../platform/http_thread_callback.hpp" #include "../../../../../std/string.hpp" +#include "../../../../../base/logging.hpp" + +#include "../maps/DownloadUI.hpp" +#include "../jni/jni_thread.hpp" // @TODO empty stub, add android implementation +HttpThread::HttpThread(string const & url, + downloader::IHttpThreadCallback & cb, + int64_t beg, + int64_t end, + int64_t size, + string const & pb) +{ + LOG(LINFO, ("creating httpThread: ", &cb, url, beg, end, size, pb)); + + /// should create java object here. + JNIEnv * env = jni::GetCurrentThreadJNIEnv(); + + LOG(LINFO, ("env : ", env)); + + jclass k = env->FindClass("com/mapswithme/maps/downloader/DownloadChunkTask"); + + jni::Method ctor(k, "", "(JLjava/lang/String;JJJLjava/lang/String;)V"); + + jlong _id = reinterpret_cast(&cb); + jlong _beg = static_cast(beg); + jlong _end = static_cast(end); + jlong _size = static_cast(size); + jstring _url = env->NewStringUTF(url.c_str()); + jstring _pb = env->NewStringUTF(pb.c_str()); + + m_self = env->NewObject(k, ctor.GetMethodID(), _id, _url, _beg, _end, _size, _pb); + + LOG(LINFO, ("starting a newly created thread", m_self)); + + jni::Method startFn(k, "start", "()V"); + + startFn.CallVoid(m_self); + + LOG(LINFO, ("started separate download thread")); +} + +HttpThread::~HttpThread() +{ + LOG(LINFO, ("destroying http_thread")); + JNIEnv * env = jni::GetCurrentThreadJNIEnv(); + + jclass k = env->FindClass("com/mapswithme/maps/downloader/DownloadChunkTask"); + jni::Method cancelFn(k, "cancel", "(Z)V"); + cancelFn.CallVoid(m_self, true); + + env->DeleteLocalRef(m_self); +} + namespace downloader { -HttpThread * CreateNativeHttpThread(string const & url, - downloader::IHttpThreadCallback & cb, - int64_t beg, - int64_t end, - int64_t size, - string const & pb) -{ - return 0; -} -void DeleteNativeHttpThread(HttpThread * request) -{ -} + HttpThread * CreateNativeHttpThread(string const & url, + downloader::IHttpThreadCallback & cb, + int64_t beg, + int64_t end, + int64_t size, + string const & pb) + { + return new HttpThread(url, cb, beg, end, size, pb); + } + + void DeleteNativeHttpThread(HttpThread * request) + { + delete request; + } } // namespace downloader + +extern "C" +{ + JNIEXPORT void JNICALL + Java_com_mapswithme_maps_downloader_DownloadChunkTask_onWrite(JNIEnv * env, jobject thiz, + jlong httpCallbackID, jlong beg, jcharArray data, jlong size) + { + LOG(LINFO, ("onWrite: ", beg, size)); + downloader::IHttpThreadCallback * cb = reinterpret_cast(httpCallbackID); + JNIEnv * env0 = jni::GetCurrentThreadJNIEnv(); + jchar * buf = env0->GetCharArrayElements(data, 0); + cb->OnWrite(beg, buf, size); + env0->ReleaseCharArrayElements(data, buf, 0); + } + + JNIEXPORT void JNICALL + Java_com_mapswithme_maps_downloader_DownloadChunkTask_onFinish(JNIEnv * env, jobject thiz, + jlong httpCallbackID, jlong httpCode, jlong beg, jlong end) + { + LOG(LINFO, ("onFinish: ", httpCode, beg, end)); + downloader::IHttpThreadCallback * cb = reinterpret_cast(httpCallbackID); + cb->OnFinish(httpCode, beg, end); + } +} diff --git a/android/jni/com/mapswithme/platform/http_thread_android.hpp b/android/jni/com/mapswithme/platform/http_thread_android.hpp index 9a188786ea..d2a4db56e3 100644 --- a/android/jni/com/mapswithme/platform/http_thread_android.hpp +++ b/android/jni/com/mapswithme/platform/http_thread_android.hpp @@ -1,7 +1,23 @@ #pragma once #include "../../../../../std/stdint.hpp" +#include "../../../../../std/string.hpp" +#include "../../../../../platform/http_thread_callback.hpp" +#include class HttpThread { +private: + + jobject m_self; + +public: + + HttpThread(string const & url, + downloader::IHttpThreadCallback & cb, + int64_t beg, + int64_t end, + int64_t size, + string const & pb); + ~HttpThread(); }; diff --git a/android/src/com/mapswithme/maps/DownloadUI.java b/android/src/com/mapswithme/maps/DownloadUI.java index e0c2268014..42c7a53f80 100644 --- a/android/src/com/mapswithme/maps/DownloadUI.java +++ b/android/src/com/mapswithme/maps/DownloadUI.java @@ -7,6 +7,7 @@ import android.preference.CheckBoxPreference; import android.preference.Preference; import android.preference.PreferenceActivity; import android.preference.PreferenceScreen; +import android.util.Log; public class DownloadUI extends PreferenceActivity { @@ -14,8 +15,12 @@ public class DownloadUI extends PreferenceActivity private native int countriesCount(int group, int country, int region); private native int countryStatus(int group, int country, int region); - private native long countrySizeInBytes(int group, int country, int region); + private native long countryLocalSizeInBytes(int group, int country, int region); + private native long countryRemoteSizeInBytes(int group, int country, int region); private native String countryName(int group, int country, int region); + private native void nativeCreate(); + private native void nativeDestroy(); + private native void downloadCountry(int group, int country, int region); @Override protected void onCreate(Bundle savedInstanceState) @@ -24,6 +29,28 @@ public class DownloadUI extends PreferenceActivity // Root PreferenceScreen root = getPreferenceManager().createPreferenceScreen(this); setPreferenceScreen(createCountriesHierarchy(root, -1, -1, -1)); + + nativeCreate(); + } + + @Override + public void onDestroy() + { + super.onDestroy(); + + nativeDestroy(); + } + + public void onChangeCountry(int group, int country, int region) + { + Log.d(TAG, new StringBuilder("onChangeCountry %1, %2, %3").append(group).append(country).append(region).toString()); + /// Should post a message onto gui thread as it could be called from the HttpThread + } + + public void onProgress(int group, int country, int region, long p1, long p2) + { + Log.d(TAG, new StringBuilder("onProgress %1, %2, %3, %4, %5").append(group).append(country).append(region).append(p1).append(p2).toString()); + /// Should post a message onto gui thread as it could be called from the HttpThread } private Preference createElement(int group, int country, int region) @@ -35,8 +62,10 @@ public class DownloadUI extends PreferenceActivity CheckBoxPreference c = new CheckBoxPreference(this); c.setKey(group + " " + country + " " + region); c.setTitle(name); - final long s = countrySizeInBytes(group, country, region); - final long remoteBytes = (s & 0xffff); + + final long localBytes = countryLocalSizeInBytes(group, country, region); + final long remoteBytes = countryRemoteSizeInBytes(group, country, region); + final String sizeString; if (remoteBytes > 1024 * 1024) sizeString = remoteBytes / (1024 * 1024) + "Mb"; @@ -100,10 +129,15 @@ public class DownloadUI extends PreferenceActivity final int country = Integer.parseInt(keys[1]); final int region = Integer.parseInt(keys[2]); - if (((CheckBoxPreference)preference).isChecked()) +/* switch (countryStatus(group, country, region)) { - // - } + case 0: //EOnDisk + { + /// Ask about deleting + }*/ + downloadCountry(group, country, region); + + Log.d(TAG, "started country download"); } return super.onPreferenceTreeClick(preferenceScreen, preference); } diff --git a/android/src/com/mapswithme/maps/downloader/DownloadChunkTask.java b/android/src/com/mapswithme/maps/downloader/DownloadChunkTask.java new file mode 100644 index 0000000000..e83844dc15 --- /dev/null +++ b/android/src/com/mapswithme/maps/downloader/DownloadChunkTask.java @@ -0,0 +1,136 @@ +package com.mapswithme.maps.downloader; + +import java.io.BufferedReader; +import java.io.DataInputStream; +import java.io.DataOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.net.HttpURLConnection; +import java.net.MalformedURLException; +import java.net.URL; +import android.util.Log; + +public class DownloadChunkTask extends Thread +{ + private final String TAG = "DownloadChunkTask"; + + private long m_httpCallbackID; + private String m_url; + private long m_beg; + private long m_end; + private long m_size; + private String m_postBody; + + public DownloadChunkTask(long httpCallbackID, String url, long beg, long end, long size, String postBody) + { + Log.d(TAG, "creating new task: " + httpCallbackID + url + beg + end + size + postBody); + + m_httpCallbackID = httpCallbackID; + m_url = url; + m_beg = beg; + m_end = end; + m_size = size; + m_postBody = postBody; + } + + public void debugPause() + { + try + { + sleep(1000); + } + catch (Exception ex) + {} + } + + @Override + public void run() + { + Log.d(TAG, ("running DownloadChunkTask in separate thread")); + + URL url; + try + { + url = new URL(m_url); + + Log.d(TAG, ("opening connection to " + m_url)); + + HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection(); + + Log.d(TAG, ("configuring connection parameter")); + + urlConnection.setDoOutput(true); + urlConnection.setChunkedStreamingMode(0); + urlConnection.setRequestProperty("Content-Type", "application/json"); + urlConnection.setUseCaches(false); + + if (m_beg != -1) + { + Log.d(TAG, ("setting requested range")); + + if (m_end != -1) + urlConnection.setRequestProperty("Range", new StringBuilder("bytes=%1-%2").append(m_beg).append(m_end).toString()); + else + urlConnection.setRequestProperty("Range", new StringBuilder("bytes=%1-").append(m_beg).toString()); + } + + if (!m_postBody.isEmpty()) + { + Log.d(TAG, ("writing POST body")); + + DataOutputStream os = new DataOutputStream(urlConnection.getOutputStream()); + os.writeChars(m_postBody); + } + + Log.d(TAG, ("getting response")); + + int response = urlConnection.getResponseCode(); + if (response == HttpURLConnection.HTTP_OK) + { + Log.d(TAG, "saving downloaded data"); + + BufferedReader reader = new BufferedReader(new InputStreamReader(urlConnection.getInputStream())); + + int chunkSize = 1024 * 64; + int offs = 0; + + char [] data = new char[chunkSize]; + + while (true) + { + long readBytes = reader.read(data); + + Log.d(TAG, "got " + readBytes + " bytes of data"); + + if (readBytes == -1) + { + onFinish(m_httpCallbackID, 200, m_beg, m_end); + break; + } + else + { + onWrite(m_httpCallbackID, m_beg + offs, data, readBytes); + offs += readBytes; + } + } + } + else + { + onFinish(m_httpCallbackID, response, m_beg, m_end); + } + } + catch (MalformedURLException ex) + { + Log.d(TAG, "invalid url : " + m_url); + } + catch (IOException ex) + { + Log.d(TAG, "ioexception : " + ex.toString()); + /// report error here + } + } + + private native void onWrite(long httpCallbackID, long beg, char [] data, long size); + private native void onFinish(long httpCallbackID, long httpCode, long beg, long end); +} diff --git a/storage/storage.hpp b/storage/storage.hpp index d2f8b9df09..deb4dc78d1 100644 --- a/storage/storage.hpp +++ b/storage/storage.hpp @@ -17,7 +17,7 @@ namespace storage /// Used in GUI enum TStatus { - EOnDisk, + EOnDisk = 0, ENotDownloaded, EDownloadFailed, EDownloading,