2015-01-13 17:30:07 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2015 Chukong Technologies Inc.
|
|
|
|
|
|
|
|
http://www.cocos2d-x.org
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
|
|
|
#include "CocosPlayClient.h"
|
|
|
|
#include <map>
|
2015-05-07 10:37:19 +08:00
|
|
|
#include<unordered_map>
|
|
|
|
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
|
2015-01-13 17:30:07 +08:00
|
|
|
#include "jni/JniHelper.h"
|
|
|
|
#include "platform/CCCommon.h"
|
2015-05-07 10:37:19 +08:00
|
|
|
#include "platform/CCFileUtils.h"
|
2015-01-13 17:30:07 +08:00
|
|
|
|
|
|
|
using namespace cocos2d;
|
|
|
|
|
|
|
|
#define LOG_TAG "CocosPlayClient.cpp"
|
|
|
|
#if COCOS2D_DEBUG
|
2015-01-15 18:21:03 +08:00
|
|
|
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__)
|
|
|
|
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,LOG_TAG,__VA_ARGS__)
|
|
|
|
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)
|
2015-01-13 17:30:07 +08:00
|
|
|
#else
|
|
|
|
#define LOGD(...)
|
2015-01-15 18:21:03 +08:00
|
|
|
#define LOGW(...)
|
|
|
|
#define LOGE(...)
|
2015-01-13 17:30:07 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static std::string __gameRootPath;
|
|
|
|
static std::unordered_map<std::string, bool> __fileExistsCaches;
|
|
|
|
static bool __isCocosPlayInited = false;
|
|
|
|
static bool __isCocosPlayEnabled = false;
|
|
|
|
static bool __isDemo = false;
|
|
|
|
static bool __isNotifyFileLoadedEnabled = false;
|
|
|
|
static jobject __classLoader;
|
|
|
|
static jmethodID __findClassMethod;
|
|
|
|
static pthread_key_t __threadKey;
|
|
|
|
|
|
|
|
|
|
|
|
#define COCOSPLAYCLIENT_CLASS_NAME "com/chukong/cocosplay/client/CocosPlayClient"
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
jobjectArray Java_com_chukong_cocosplay_client_CocosPlayClient_getSearchPaths(JNIEnv* env, jobject thiz)
|
|
|
|
{
|
|
|
|
auto stringClass = env->FindClass("java/lang/String");
|
|
|
|
auto& paths = cocos2d::FileUtils::getInstance()->getSearchPaths();
|
|
|
|
auto count = paths.size();
|
|
|
|
auto pathArray = env->NewObjectArray(count, stringClass, 0);
|
|
|
|
for (int i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
env->SetObjectArrayElement(pathArray, i, env->NewStringUTF(paths[i].c_str()));
|
|
|
|
}
|
|
|
|
|
|
|
|
return pathArray;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace cocosplay {
|
|
|
|
|
|
|
|
static void detachCurrentThread(void *env) {
|
|
|
|
JniHelper::getJavaVM()->DetachCurrentThread();
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool getEnv(JNIEnv **env)
|
|
|
|
{
|
|
|
|
bool bRet = false;
|
|
|
|
|
|
|
|
switch(JniHelper::getJavaVM()->GetEnv((void**)env, JNI_VERSION_1_4))
|
|
|
|
{
|
|
|
|
case JNI_OK:
|
|
|
|
bRet = true;
|
|
|
|
break;
|
|
|
|
case JNI_EDETACHED:
|
|
|
|
pthread_key_create (&__threadKey, detachCurrentThread);
|
|
|
|
if (JniHelper::getJavaVM()->AttachCurrentThread(env, 0) < 0)
|
|
|
|
{
|
|
|
|
LOGD("%s", "Failed to get the environment using AttachCurrentThread()");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (pthread_getspecific(__threadKey) == NULL) {
|
|
|
|
pthread_setspecific(__threadKey, env);
|
|
|
|
}
|
|
|
|
bRet = true;
|
|
|
|
break;
|
|
|
|
default:
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGE("%s", "Failed to get the environment using GetEnv()");
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bRet;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void initClassLoaderForMultiThread()
|
|
|
|
{
|
|
|
|
JNIEnv *env = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (! getEnv(&env))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
jclass cocos2dClass = env->FindClass(COCOSPLAYCLIENT_CLASS_NAME);
|
|
|
|
if(env->ExceptionCheck())
|
|
|
|
{
|
|
|
|
env->ExceptionDescribe();
|
|
|
|
env->ExceptionClear();
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("Exception initClassLoaderForMultiThread cocos2dClass is exception");
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
jclass classClass = env->GetObjectClass(cocos2dClass);
|
|
|
|
if(env->ExceptionCheck())
|
|
|
|
{
|
|
|
|
env->ExceptionDescribe();
|
|
|
|
env->ExceptionClear();
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("Exception initClassLoaderForMultiThread classClass is exception");
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
jclass classLoaderClass = env->FindClass("java/lang/ClassLoader");
|
|
|
|
if(env->ExceptionCheck())
|
|
|
|
{
|
|
|
|
env->ExceptionDescribe();
|
|
|
|
env->ExceptionClear();
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("Exception initClassLoaderForMultiThread classLoaderClass");
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
jmethodID getClassLoaderMethod = env->GetMethodID(classClass,"getClassLoader","()Ljava/lang/ClassLoader;");
|
|
|
|
jobject classLoader = env->CallObjectMethod(cocos2dClass, getClassLoaderMethod);
|
|
|
|
if(env->ExceptionCheck())
|
|
|
|
{
|
|
|
|
env->ExceptionDescribe();
|
|
|
|
env->ExceptionClear();
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("Exception initClassLoaderForMultiThread classLoader");
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
__classLoader = env->NewGlobalRef(classLoader);
|
|
|
|
jmethodID findClassMethod = env->GetMethodID(classLoaderClass, "findClass", "(Ljava/lang/String;)Ljava/lang/Class;");
|
|
|
|
__findClassMethod = findClassMethod;
|
|
|
|
|
|
|
|
if(env->ExceptionCheck())
|
|
|
|
{
|
|
|
|
env->ExceptionDescribe();
|
|
|
|
env->ExceptionClear();
|
|
|
|
__findClassMethod = NULL;
|
|
|
|
__classLoader = NULL;
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("Exception initClassLoaderForMultiThread findClassMethod");
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}while(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static jclass getClassID_(const char *className, JNIEnv *env)
|
|
|
|
{
|
|
|
|
JNIEnv *pEnv = env;
|
|
|
|
jclass ret = 0;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (! pEnv)
|
|
|
|
{
|
|
|
|
if (! getEnv(&pEnv))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = pEnv->FindClass(className);
|
|
|
|
if (! ret)
|
|
|
|
{
|
|
|
|
if(__classLoader)
|
|
|
|
{
|
|
|
|
pEnv->ExceptionClear();
|
|
|
|
jstring jstrName = (pEnv)->NewStringUTF(className);
|
|
|
|
ret = (jclass)pEnv->CallObjectMethod(__classLoader, __findClassMethod, jstrName);
|
|
|
|
pEnv->DeleteLocalRef(jstrName);
|
|
|
|
if(ret) break;
|
|
|
|
}
|
|
|
|
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGE("Failed to find class of %s", className);
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool getStaticMethodInfo(cocos2d::JniMethodInfo &methodinfo, const char *className, const char *methodName, const char *paramCode)
|
|
|
|
{
|
|
|
|
jmethodID methodID = 0;
|
|
|
|
JNIEnv *pEnv = 0;
|
|
|
|
bool bRet = false;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (! getEnv(&pEnv))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
jclass classID = getClassID_(className, pEnv);
|
|
|
|
if(!classID) break;
|
|
|
|
|
|
|
|
methodID = pEnv->GetStaticMethodID(classID, methodName, paramCode);
|
|
|
|
if (! methodID)
|
|
|
|
{
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("Failed to find static method id of %s", methodName);
|
2015-01-13 17:30:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
methodinfo.classID = classID;
|
|
|
|
methodinfo.env = pEnv;
|
|
|
|
methodinfo.methodID = methodID;
|
|
|
|
|
|
|
|
bRet = true;
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
return bRet;
|
|
|
|
}
|
|
|
|
|
|
|
|
void lazyInit()
|
|
|
|
{
|
|
|
|
if (__isCocosPlayInited)
|
|
|
|
return;
|
|
|
|
|
|
|
|
JniMethodInfo t;
|
|
|
|
if (JniHelper::getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "isEnabled", "()Z"))
|
|
|
|
{
|
|
|
|
__isCocosPlayEnabled = t.env->CallStaticBooleanMethod(t.classID, t.methodID);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
LOGD("isEnabled = %d", __isCocosPlayEnabled);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (JniHelper::getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "isDemo", "()Z"))
|
|
|
|
{
|
|
|
|
__isDemo = t.env->CallStaticBooleanMethod(t.classID, t.methodID);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
LOGD("isDemo = %d",__isDemo);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (JniHelper::getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "isNotifyFileLoadedEnabled", "()Z"))
|
|
|
|
{
|
|
|
|
__isNotifyFileLoadedEnabled = t.env->CallStaticBooleanMethod(t.classID, t.methodID);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
LOGD("isNotifyFileLoadedEnabled = %d", __isNotifyFileLoadedEnabled);
|
|
|
|
}
|
|
|
|
|
2015-01-15 18:21:03 +08:00
|
|
|
if (__isCocosPlayEnabled)
|
|
|
|
{
|
|
|
|
initClassLoaderForMultiThread();
|
|
|
|
}
|
2015-01-13 17:30:07 +08:00
|
|
|
|
|
|
|
__isCocosPlayInited = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEnabled()
|
|
|
|
{
|
|
|
|
return __isCocosPlayEnabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isDemo()
|
|
|
|
{
|
|
|
|
return __isDemo;
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateAssets(const std::string& filePath)
|
|
|
|
{
|
|
|
|
if (!__isCocosPlayInited)
|
|
|
|
{
|
|
|
|
lazyInit();
|
|
|
|
}
|
|
|
|
|
2015-01-15 18:21:03 +08:00
|
|
|
if (!__isCocosPlayEnabled || __isDemo)
|
2015-01-13 17:30:07 +08:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!fileExists(filePath))
|
|
|
|
{
|
|
|
|
LOGD("file ( %s ) doesn't exist, updateAssets cancelled", filePath.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
JniMethodInfo t;
|
|
|
|
if (getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "updateAssets", "(Ljava/lang/String;)V"))
|
|
|
|
{
|
|
|
|
jstring stringArg = t.env->NewStringUTF(filePath.c_str());
|
|
|
|
t.env->CallStaticVoidMethod(t.classID, t.methodID, stringArg);
|
|
|
|
t.env->DeleteLocalRef(stringArg);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
}
|
|
|
|
|
|
|
|
LOGD("updateAssets (%s) OK!", filePath.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fileExists(const std::string& filePath)
|
|
|
|
{
|
|
|
|
auto iter = __fileExistsCaches.find(filePath);
|
|
|
|
if (iter != __fileExistsCaches.end())
|
|
|
|
{
|
|
|
|
LOGD("Return file path ( %s ) in cache!", filePath.c_str());
|
|
|
|
if(!iter->second)
|
|
|
|
{
|
|
|
|
auto fp = fopen(filePath.c_str(), "r");
|
|
|
|
if (fp)
|
|
|
|
{
|
|
|
|
iter->second = true;
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return iter->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ret = false;
|
|
|
|
JniMethodInfo t;
|
|
|
|
if (getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "fileExists", "(Ljava/lang/String;)Z"))
|
|
|
|
{
|
|
|
|
jstring stringArg = t.env->NewStringUTF(filePath.c_str());
|
|
|
|
ret = t.env->CallStaticBooleanMethod(t.classID, t.methodID, stringArg);
|
|
|
|
t.env->DeleteLocalRef(stringArg);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
}
|
|
|
|
|
|
|
|
__fileExistsCaches[filePath] = ret;
|
|
|
|
|
|
|
|
LOGD("fileExists return (%d), path (%s)!", ret, filePath.c_str());
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void notifyFileLoaded(const std::string& filePath)
|
|
|
|
{
|
|
|
|
if (!__isNotifyFileLoadedEnabled)
|
|
|
|
return;
|
|
|
|
|
|
|
|
JniMethodInfo t;
|
|
|
|
if (getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "notifyFileLoaded", "(Ljava/lang/String;)V"))
|
|
|
|
{
|
|
|
|
jstring stringArg = t.env->NewStringUTF(filePath.c_str());
|
|
|
|
t.env->CallStaticVoidMethod(t.classID, t.methodID, stringArg);
|
|
|
|
t.env->DeleteLocalRef(stringArg);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string getGameRoot()
|
|
|
|
{
|
|
|
|
if (!__isCocosPlayEnabled)
|
|
|
|
{
|
2015-01-15 18:21:03 +08:00
|
|
|
LOGW("CocosPlayClient isn't enabled!");
|
2015-01-13 17:30:07 +08:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (__gameRootPath.empty())
|
|
|
|
{
|
|
|
|
JniMethodInfo t;
|
|
|
|
if (JniHelper::getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "getGameRoot", "()Ljava/lang/String;"))
|
|
|
|
{
|
|
|
|
jstring str = (jstring)t.env->CallStaticObjectMethod(t.classID, t.methodID);
|
|
|
|
__gameRootPath = JniHelper::jstring2string(str);
|
|
|
|
t.env->DeleteLocalRef(str);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
}
|
|
|
|
LOGD("GameRoot : %s", __gameRootPath.c_str());
|
|
|
|
}
|
|
|
|
return __gameRootPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
void purgeCachedEntries()
|
|
|
|
{
|
|
|
|
__fileExistsCaches.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void purgeCachedByFile(const std::string& filePath)
|
|
|
|
{
|
|
|
|
__fileExistsCaches.erase(filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
void notifyDemoEnded()
|
|
|
|
{
|
|
|
|
JniMethodInfo t;
|
|
|
|
if (JniHelper::getStaticMethodInfo(t, COCOSPLAYCLIENT_CLASS_NAME, "notifyDemoEnded", "()V"))
|
|
|
|
{
|
|
|
|
t.env->CallStaticVoidMethod(t.classID, t.methodID);
|
|
|
|
t.env->DeleteLocalRef(t.classID);
|
|
|
|
LOGD("Game demo was ended!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace cocosplay {
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
namespace cocosplay {
|
|
|
|
|
|
|
|
bool isEnabled() { return false; }
|
|
|
|
bool isDemo() { return false; }
|
|
|
|
void updateAssets(const std::string& filePath) {}
|
|
|
|
bool fileExists(const std::string& filePath) { return false; }
|
|
|
|
void notifyFileLoaded(const std::string& filePath) {}
|
|
|
|
std::string getGameRoot() { return ""; }
|
|
|
|
void purgeCachedEntries() {}
|
|
|
|
void notifyDemoEnded() {}
|
|
|
|
void purgeCachedByFile(const std::string& filePath){}
|
|
|
|
} // namespace cocosplay {
|
|
|
|
|
|
|
|
#endif
|