/* * @Description: * @Author: ekibun * @Date: 2020-08-16 11:08:23 * @LastEditors: ekibun * @LastEditTime: 2020-08-20 11:29:32 */ #include #include #include "jni_helper.hpp" #include "../../../../cxx/js_engine.hpp" namespace std { template <> struct hash { std::size_t operator()(const qjs::Value &key) const { return JS_VALUE_GET_TAG(key.v); } }; } // namespace std namespace qjs { JSValue javaToJs(JSContext *ctx, JNIEnv *env, jobject val, std::unordered_map cache = std::unordered_map()) { if (val == nullptr) return JS_UNDEFINED; if (cache.find(val) != cache.end()) return cache[val]; jclass objclass = env->GetObjectClass(val); jclass classclass = env->GetObjectClass((jobject)objclass); jmethodID mid = env->GetMethodID(classclass, "getName", "()Ljava/lang/String;"); jobject clsObj = env->CallObjectMethod(objclass, mid); std::string className(env->GetStringUTFChars((jstring)clsObj, 0)); if (className.compare("[B") == 0) { jsize len = env->GetArrayLength((jbyteArray)val); return JS_NewArrayBufferCopy(ctx, (uint8_t *)env->GetByteArrayElements((jbyteArray)val, 0), len); } else if (className.compare("[I") == 0) { jsize len = env->GetArrayLength((jintArray)val); return JS_NewArrayBufferCopy(ctx, (uint8_t *)env->GetIntArrayElements((jintArray)val, 0), len * 4); } else if (className.compare("[J") == 0) { jsize len = env->GetArrayLength((jlongArray)val); return JS_NewArrayBufferCopy(ctx, (uint8_t *)env->GetLongArrayElements((jlongArray)val, 0), len * 8); } else if (className.compare("[D") == 0) { jsize size = env->GetArrayLength((jdoubleArray)val); JSValue array = JS_NewArray(ctx); auto buf = env->GetDoubleArrayElements((jdoubleArray)val, 0); for (uint32_t i = 0; i < size; i++) JS_DefinePropertyValue( ctx, array, JS_NewAtomUInt32(ctx, i), JS_NewFloat64(ctx, buf[i]), JS_PROP_C_W_E); return array; } else if (className.compare("java.lang.Boolean") == 0) { jmethodID getVal = env->GetMethodID(objclass, "booleanValue", "()Z"); return JS_NewBool(ctx, env->CallBooleanMethod(val, getVal)); } else if (className.compare("java.lang.Integer") == 0) { jmethodID getVal = env->GetMethodID(objclass, "intValue", "()I"); return JS_NewInt32(ctx, env->CallIntMethod(val, getVal)); } else if (className.compare("java.lang.Long") == 0) { jmethodID getVal = env->GetMethodID(objclass, "longValue", "()J"); return JS_NewInt64(ctx, env->CallLongMethod(val, getVal)); } else if (className.compare("java.lang.Double") == 0) { jmethodID getVal = env->GetMethodID(objclass, "doubleValue", "()D"); return JS_NewFloat64(ctx, env->CallDoubleMethod(val, getVal)); } else if (className.compare("java.lang.String") == 0) { return JS_NewString(ctx, env->GetStringUTFChars((jstring)val, 0)); } else if (className.compare("java.util.ArrayList") == 0) { jobjectArray list = jniToArray(env, val); jsize size = env->GetArrayLength(list); JSValue array = JS_NewArray(ctx); cache[val] = array; for (uint32_t i = 0; i < size; i++) JS_DefinePropertyValue( ctx, array, JS_NewAtomUInt32(ctx, i), javaToJs(ctx, env, env->GetObjectArrayElement(list, i), cache), JS_PROP_C_W_E); return array; } else if (className.compare("java.util.HashMap") == 0) { // 获取HashMap类entrySet()方法ID jmethodID entrySetMID = env->GetMethodID(objclass, "entrySet", "()Ljava/util/Set;"); // 调用entrySet()方法获取Set对象 jobject setObj = env->CallObjectMethod(val, entrySetMID); // 获取Set类中iterator()方法ID jclass setClass = env->FindClass("java/util/Set"); jmethodID iteratorMID = env->GetMethodID(setClass, "iterator", "()Ljava/util/Iterator;"); // 调用iterator()方法获取Iterator对象 jobject iteratorObj = env->CallObjectMethod(setObj, iteratorMID); // 获取Iterator类中hasNext()方法ID // 用于while循环判断HashMap中是否还有数据 jclass iteratorClass = env->FindClass("java/util/Iterator"); jmethodID hasNextMID = env->GetMethodID(iteratorClass, "hasNext", "()Z"); // 获取Iterator类中next()方法ID // 用于读取HashMap中的每一条数据 jmethodID nextMID = env->GetMethodID(iteratorClass, "next", "()Ljava/lang/Object;"); // 获取Map.Entry类中getKey()和getValue()的方法ID // 用于读取“K-V”键值对,注意:内部类使用$符号表示 jclass entryClass = env->FindClass("java/util/Map$Entry"); jmethodID getKeyMID = env->GetMethodID(entryClass, "getKey", "()Ljava/lang/Object;"); jmethodID getValueMID = env->GetMethodID(entryClass, "getValue", "()Ljava/lang/Object;"); // JSObject JSValue obj = JS_NewObject(ctx); cache[val] = obj; // 循环检测HashMap中是否还有数据 while (env->CallBooleanMethod(iteratorObj, hasNextMID)) { // 读取一条数据 jobject entryObj = env->CallObjectMethod(iteratorObj, nextMID); JS_DefinePropertyValue( ctx, obj, JS_ValueToAtom(ctx, javaToJs(ctx, env, env->CallObjectMethod(entryObj, getKeyMID), cache)), javaToJs(ctx, env, env->CallObjectMethod(entryObj, getValueMID), cache), JS_PROP_C_W_E); } return obj; } return JS_UNDEFINED; } jobject jsToJava(JNIEnv *env, qjs::Value val, std::unordered_map cache = std::unordered_map()) { int tag = JS_VALUE_GET_TAG(val.v); if (JS_TAG_IS_FLOAT64(tag)) return jniWrapPrimity(env, (double)val); switch (tag) { case JS_TAG_BOOL: return jniWrapPrimity(env, (bool)val); case JS_TAG_INT: return jniWrapPrimity(env, (int64_t)val); case JS_TAG_STRING: return env->NewStringUTF(((std::string)val).c_str()); case JS_TAG_OBJECT: { // ArrayBuffer size_t size; uint8_t *buf = JS_GetArrayBuffer(val.ctx, &size, val.v); if (buf) { jbyteArray arr = env->NewByteArray(size); env->SetByteArrayRegion(arr, 0, size, (int8_t *)buf); return arr; } } if (cache.find(val) != cache.end()) return cache[val]; if (JS_IsFunction(val.ctx, val.v)) { std::map retMap; retMap[env->NewStringUTF("__js_function__")] = jniWrapPrimity(env, (int64_t) new JSValue{JS_DupValue(val.ctx, val.v)}); return jniWrapMap(env, retMap); } else if (JS_IsArray(val.ctx, val.v) > 0) { uint32_t arrlen = (uint32_t)val["length"]; jclass class_arraylist = env->FindClass("java/util/ArrayList"); jmethodID arraylist_init = env->GetMethodID(class_arraylist, "", "()V"); jobject list = env->NewObject(class_arraylist, arraylist_init); jmethodID arraylist_add = env->GetMethodID(class_arraylist, "add", "(Ljava/lang/Object;)Z"); for (uint32_t i = 0; i < arrlen; i++) { env->CallBooleanMethod(list, arraylist_add, jsToJava(env, val[i], cache)); } cache[val] = list; return list; } else { qjs::JSPropertyEnum *ptab; uint32_t plen; if (JS_GetOwnPropertyNames(val.ctx, &ptab, &plen, val.v, -1)) return nullptr; std::map retMap; for (uint32_t i = 0; i < plen; i++) { retMap[jsToJava(env, {val.ctx, JS_AtomToValue(val.ctx, ptab[i].atom)}, cache)] = jsToJava(env, {val.ctx, JS_GetProperty(val.ctx, val.v, ptab[i].atom)}, cache); JS_FreeAtom(val.ctx, ptab[i].atom); } js_free(val.ctx, ptab); jobject ret = jniWrapMap(env, retMap); cache[val] = ret; return ret; } default: return nullptr; } } } // namespace qjs