java_godot_wrapper.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. /**************************************************************************/
  2. /* java_godot_wrapper.cpp */
  3. /**************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /**************************************************************************/
  8. /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
  9. /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /**************************************************************************/
  30. #include "java_godot_wrapper.h"
  31. // JNIEnv is only valid within the thread it belongs to, in a multi threading environment
  32. // we can't cache it.
  33. // For Godot we call most access methods from our thread and we thus get a valid JNIEnv
  34. // from get_jni_env(). For one or two we expect to pass the environment
  35. // TODO we could probably create a base class for this...
  36. GodotJavaWrapper::GodotJavaWrapper(JNIEnv *p_env, jobject p_activity, jobject p_godot_instance) {
  37. godot_instance = p_env->NewGlobalRef(p_godot_instance);
  38. activity = p_env->NewGlobalRef(p_activity);
  39. // get info about our Godot class so we can get pointers and stuff...
  40. godot_class = p_env->FindClass("org/godotengine/godot/Godot");
  41. if (godot_class) {
  42. godot_class = (jclass)p_env->NewGlobalRef(godot_class);
  43. } else {
  44. // this is a pretty serious fail.. bail... pointers will stay 0
  45. return;
  46. }
  47. activity_class = p_env->FindClass("android/app/Activity");
  48. if (activity_class) {
  49. activity_class = (jclass)p_env->NewGlobalRef(activity_class);
  50. } else {
  51. // this is a pretty serious fail.. bail... pointers will stay 0
  52. return;
  53. }
  54. // get some Godot method pointers...
  55. _restart = p_env->GetMethodID(godot_class, "restart", "()V");
  56. _finish = p_env->GetMethodID(godot_class, "forceQuit", "(I)Z");
  57. _set_keep_screen_on = p_env->GetMethodID(godot_class, "setKeepScreenOn", "(Z)V");
  58. _alert = p_env->GetMethodID(godot_class, "alert", "(Ljava/lang/String;Ljava/lang/String;)V");
  59. _is_dark_mode_supported = p_env->GetMethodID(godot_class, "isDarkModeSupported", "()Z");
  60. _is_dark_mode = p_env->GetMethodID(godot_class, "isDarkMode", "()Z");
  61. _get_clipboard = p_env->GetMethodID(godot_class, "getClipboard", "()Ljava/lang/String;");
  62. _set_clipboard = p_env->GetMethodID(godot_class, "setClipboard", "(Ljava/lang/String;)V");
  63. _has_clipboard = p_env->GetMethodID(godot_class, "hasClipboard", "()Z");
  64. _request_permission = p_env->GetMethodID(godot_class, "requestPermission", "(Ljava/lang/String;)Z");
  65. _request_permissions = p_env->GetMethodID(godot_class, "requestPermissions", "()Z");
  66. _get_granted_permissions = p_env->GetMethodID(godot_class, "getGrantedPermissions", "()[Ljava/lang/String;");
  67. _get_ca_certificates = p_env->GetMethodID(godot_class, "getCACertificates", "()Ljava/lang/String;");
  68. _init_input_devices = p_env->GetMethodID(godot_class, "initInputDevices", "()V");
  69. _vibrate = p_env->GetMethodID(godot_class, "vibrate", "(II)V");
  70. _get_input_fallback_mapping = p_env->GetMethodID(godot_class, "getInputFallbackMapping", "()Ljava/lang/String;");
  71. _on_godot_setup_completed = p_env->GetMethodID(godot_class, "onGodotSetupCompleted", "()V");
  72. _on_godot_main_loop_started = p_env->GetMethodID(godot_class, "onGodotMainLoopStarted", "()V");
  73. _on_godot_terminating = p_env->GetMethodID(godot_class, "onGodotTerminating", "()V");
  74. _create_new_godot_instance = p_env->GetMethodID(godot_class, "createNewGodotInstance", "([Ljava/lang/String;)I");
  75. _get_render_view = p_env->GetMethodID(godot_class, "getRenderView", "()Lorg/godotengine/godot/GodotRenderView;");
  76. _begin_benchmark_measure = p_env->GetMethodID(godot_class, "nativeBeginBenchmarkMeasure", "(Ljava/lang/String;Ljava/lang/String;)V");
  77. _end_benchmark_measure = p_env->GetMethodID(godot_class, "nativeEndBenchmarkMeasure", "(Ljava/lang/String;Ljava/lang/String;)V");
  78. _dump_benchmark = p_env->GetMethodID(godot_class, "nativeDumpBenchmark", "(Ljava/lang/String;)V");
  79. _get_gdextension_list_config_file = p_env->GetMethodID(godot_class, "getGDExtensionConfigFiles", "()[Ljava/lang/String;");
  80. _has_feature = p_env->GetMethodID(godot_class, "hasFeature", "(Ljava/lang/String;)Z");
  81. }
  82. GodotJavaWrapper::~GodotJavaWrapper() {
  83. if (godot_view) {
  84. delete godot_view;
  85. }
  86. JNIEnv *env = get_jni_env();
  87. ERR_FAIL_NULL(env);
  88. env->DeleteGlobalRef(godot_instance);
  89. env->DeleteGlobalRef(godot_class);
  90. env->DeleteGlobalRef(activity);
  91. env->DeleteGlobalRef(activity_class);
  92. }
  93. jobject GodotJavaWrapper::get_activity() {
  94. return activity;
  95. }
  96. GodotJavaViewWrapper *GodotJavaWrapper::get_godot_view() {
  97. if (godot_view != nullptr) {
  98. return godot_view;
  99. }
  100. if (_get_render_view) {
  101. JNIEnv *env = get_jni_env();
  102. ERR_FAIL_NULL_V(env, nullptr);
  103. jobject godot_render_view = env->CallObjectMethod(godot_instance, _get_render_view);
  104. if (!env->IsSameObject(godot_render_view, nullptr)) {
  105. godot_view = new GodotJavaViewWrapper(godot_render_view);
  106. }
  107. }
  108. return godot_view;
  109. }
  110. void GodotJavaWrapper::on_godot_setup_completed(JNIEnv *p_env) {
  111. if (_on_godot_setup_completed) {
  112. if (p_env == nullptr) {
  113. p_env = get_jni_env();
  114. }
  115. p_env->CallVoidMethod(godot_instance, _on_godot_setup_completed);
  116. }
  117. }
  118. void GodotJavaWrapper::on_godot_main_loop_started(JNIEnv *p_env) {
  119. if (_on_godot_main_loop_started) {
  120. if (p_env == nullptr) {
  121. p_env = get_jni_env();
  122. }
  123. ERR_FAIL_NULL(p_env);
  124. p_env->CallVoidMethod(godot_instance, _on_godot_main_loop_started);
  125. }
  126. }
  127. void GodotJavaWrapper::on_godot_terminating(JNIEnv *p_env) {
  128. if (_on_godot_terminating) {
  129. if (p_env == nullptr) {
  130. p_env = get_jni_env();
  131. }
  132. ERR_FAIL_NULL(p_env);
  133. p_env->CallVoidMethod(godot_instance, _on_godot_terminating);
  134. }
  135. }
  136. void GodotJavaWrapper::restart(JNIEnv *p_env) {
  137. if (_restart) {
  138. if (p_env == nullptr) {
  139. p_env = get_jni_env();
  140. }
  141. ERR_FAIL_NULL(p_env);
  142. p_env->CallVoidMethod(godot_instance, _restart);
  143. }
  144. }
  145. bool GodotJavaWrapper::force_quit(JNIEnv *p_env, int p_instance_id) {
  146. if (_finish) {
  147. if (p_env == nullptr) {
  148. p_env = get_jni_env();
  149. }
  150. ERR_FAIL_NULL_V(p_env, false);
  151. return p_env->CallBooleanMethod(godot_instance, _finish, p_instance_id);
  152. }
  153. return false;
  154. }
  155. void GodotJavaWrapper::set_keep_screen_on(bool p_enabled) {
  156. if (_set_keep_screen_on) {
  157. JNIEnv *env = get_jni_env();
  158. ERR_FAIL_NULL(env);
  159. env->CallVoidMethod(godot_instance, _set_keep_screen_on, p_enabled);
  160. }
  161. }
  162. void GodotJavaWrapper::alert(const String &p_message, const String &p_title) {
  163. if (_alert) {
  164. JNIEnv *env = get_jni_env();
  165. ERR_FAIL_NULL(env);
  166. jstring jStrMessage = env->NewStringUTF(p_message.utf8().get_data());
  167. jstring jStrTitle = env->NewStringUTF(p_title.utf8().get_data());
  168. env->CallVoidMethod(godot_instance, _alert, jStrMessage, jStrTitle);
  169. env->DeleteLocalRef(jStrMessage);
  170. env->DeleteLocalRef(jStrTitle);
  171. }
  172. }
  173. bool GodotJavaWrapper::is_dark_mode_supported() {
  174. if (_is_dark_mode_supported) {
  175. JNIEnv *env = get_jni_env();
  176. ERR_FAIL_NULL_V(env, false);
  177. return env->CallBooleanMethod(godot_instance, _is_dark_mode_supported);
  178. } else {
  179. return false;
  180. }
  181. }
  182. bool GodotJavaWrapper::is_dark_mode() {
  183. if (_is_dark_mode) {
  184. JNIEnv *env = get_jni_env();
  185. ERR_FAIL_NULL_V(env, false);
  186. return env->CallBooleanMethod(godot_instance, _is_dark_mode);
  187. } else {
  188. return false;
  189. }
  190. }
  191. bool GodotJavaWrapper::has_get_clipboard() {
  192. return _get_clipboard != nullptr;
  193. }
  194. String GodotJavaWrapper::get_clipboard() {
  195. String clipboard;
  196. if (_get_clipboard) {
  197. JNIEnv *env = get_jni_env();
  198. ERR_FAIL_NULL_V(env, String());
  199. jstring s = (jstring)env->CallObjectMethod(godot_instance, _get_clipboard);
  200. clipboard = jstring_to_string(s, env);
  201. env->DeleteLocalRef(s);
  202. }
  203. return clipboard;
  204. }
  205. String GodotJavaWrapper::get_input_fallback_mapping() {
  206. String input_fallback_mapping;
  207. if (_get_input_fallback_mapping) {
  208. JNIEnv *env = get_jni_env();
  209. ERR_FAIL_NULL_V(env, String());
  210. jstring fallback_mapping = (jstring)env->CallObjectMethod(godot_instance, _get_input_fallback_mapping);
  211. input_fallback_mapping = jstring_to_string(fallback_mapping, env);
  212. env->DeleteLocalRef(fallback_mapping);
  213. }
  214. return input_fallback_mapping;
  215. }
  216. bool GodotJavaWrapper::has_set_clipboard() {
  217. return _set_clipboard != nullptr;
  218. }
  219. void GodotJavaWrapper::set_clipboard(const String &p_text) {
  220. if (_set_clipboard) {
  221. JNIEnv *env = get_jni_env();
  222. ERR_FAIL_NULL(env);
  223. jstring jStr = env->NewStringUTF(p_text.utf8().get_data());
  224. env->CallVoidMethod(godot_instance, _set_clipboard, jStr);
  225. env->DeleteLocalRef(jStr);
  226. }
  227. }
  228. bool GodotJavaWrapper::has_has_clipboard() {
  229. return _has_clipboard != nullptr;
  230. }
  231. bool GodotJavaWrapper::has_clipboard() {
  232. if (_has_clipboard) {
  233. JNIEnv *env = get_jni_env();
  234. ERR_FAIL_NULL_V(env, false);
  235. return env->CallBooleanMethod(godot_instance, _has_clipboard);
  236. } else {
  237. return false;
  238. }
  239. }
  240. bool GodotJavaWrapper::request_permission(const String &p_name) {
  241. if (_request_permission) {
  242. JNIEnv *env = get_jni_env();
  243. ERR_FAIL_NULL_V(env, false);
  244. jstring jStrName = env->NewStringUTF(p_name.utf8().get_data());
  245. bool result = env->CallBooleanMethod(godot_instance, _request_permission, jStrName);
  246. env->DeleteLocalRef(jStrName);
  247. return result;
  248. } else {
  249. return false;
  250. }
  251. }
  252. bool GodotJavaWrapper::request_permissions() {
  253. if (_request_permissions) {
  254. JNIEnv *env = get_jni_env();
  255. ERR_FAIL_NULL_V(env, false);
  256. return env->CallBooleanMethod(godot_instance, _request_permissions);
  257. } else {
  258. return false;
  259. }
  260. }
  261. Vector<String> GodotJavaWrapper::get_granted_permissions() const {
  262. Vector<String> permissions_list;
  263. if (_get_granted_permissions) {
  264. JNIEnv *env = get_jni_env();
  265. ERR_FAIL_NULL_V(env, permissions_list);
  266. jobject permissions_object = env->CallObjectMethod(godot_instance, _get_granted_permissions);
  267. jobjectArray *arr = reinterpret_cast<jobjectArray *>(&permissions_object);
  268. jsize len = env->GetArrayLength(*arr);
  269. for (int i = 0; i < len; i++) {
  270. jstring jstr = (jstring)env->GetObjectArrayElement(*arr, i);
  271. String str = jstring_to_string(jstr, env);
  272. permissions_list.push_back(str);
  273. env->DeleteLocalRef(jstr);
  274. }
  275. }
  276. return permissions_list;
  277. }
  278. Vector<String> GodotJavaWrapper::get_gdextension_list_config_file() const {
  279. Vector<String> config_file_list;
  280. if (_get_gdextension_list_config_file) {
  281. JNIEnv *env = get_jni_env();
  282. ERR_FAIL_NULL_V(env, config_file_list);
  283. jobject config_file_list_object = env->CallObjectMethod(godot_instance, _get_gdextension_list_config_file);
  284. jobjectArray *arr = reinterpret_cast<jobjectArray *>(&config_file_list_object);
  285. jsize len = env->GetArrayLength(*arr);
  286. for (int i = 0; i < len; i++) {
  287. jstring j_config_file = (jstring)env->GetObjectArrayElement(*arr, i);
  288. String config_file = jstring_to_string(j_config_file, env);
  289. config_file_list.push_back(config_file);
  290. env->DeleteLocalRef(j_config_file);
  291. }
  292. }
  293. return config_file_list;
  294. }
  295. String GodotJavaWrapper::get_ca_certificates() const {
  296. String ca_certificates;
  297. if (_get_ca_certificates) {
  298. JNIEnv *env = get_jni_env();
  299. ERR_FAIL_NULL_V(env, String());
  300. jstring s = (jstring)env->CallObjectMethod(godot_instance, _get_ca_certificates);
  301. ca_certificates = jstring_to_string(s, env);
  302. env->DeleteLocalRef(s);
  303. }
  304. return ca_certificates;
  305. }
  306. void GodotJavaWrapper::init_input_devices() {
  307. if (_init_input_devices) {
  308. JNIEnv *env = get_jni_env();
  309. ERR_FAIL_NULL(env);
  310. env->CallVoidMethod(godot_instance, _init_input_devices);
  311. }
  312. }
  313. void GodotJavaWrapper::vibrate(int p_duration_ms, float p_amplitude) {
  314. if (_vibrate) {
  315. JNIEnv *env = get_jni_env();
  316. ERR_FAIL_NULL(env);
  317. int j_amplitude = -1.0;
  318. if (p_amplitude != -1.0) {
  319. j_amplitude = CLAMP(int(p_amplitude * 255), 1, 255);
  320. }
  321. env->CallVoidMethod(godot_instance, _vibrate, p_duration_ms, j_amplitude);
  322. }
  323. }
  324. int GodotJavaWrapper::create_new_godot_instance(const List<String> &args) {
  325. if (_create_new_godot_instance) {
  326. JNIEnv *env = get_jni_env();
  327. ERR_FAIL_NULL_V(env, 0);
  328. jobjectArray jargs = env->NewObjectArray(args.size(), env->FindClass("java/lang/String"), env->NewStringUTF(""));
  329. int i = 0;
  330. for (List<String>::ConstIterator itr = args.begin(); itr != args.end(); ++itr, ++i) {
  331. jstring j_arg = env->NewStringUTF(itr->utf8().get_data());
  332. env->SetObjectArrayElement(jargs, i, j_arg);
  333. env->DeleteLocalRef(j_arg);
  334. }
  335. return env->CallIntMethod(godot_instance, _create_new_godot_instance, jargs);
  336. } else {
  337. return 0;
  338. }
  339. }
  340. void GodotJavaWrapper::begin_benchmark_measure(const String &p_context, const String &p_label) {
  341. if (_begin_benchmark_measure) {
  342. JNIEnv *env = get_jni_env();
  343. ERR_FAIL_NULL(env);
  344. jstring j_context = env->NewStringUTF(p_context.utf8().get_data());
  345. jstring j_label = env->NewStringUTF(p_label.utf8().get_data());
  346. env->CallVoidMethod(godot_instance, _begin_benchmark_measure, j_context, j_label);
  347. env->DeleteLocalRef(j_context);
  348. env->DeleteLocalRef(j_label);
  349. }
  350. }
  351. void GodotJavaWrapper::end_benchmark_measure(const String &p_context, const String &p_label) {
  352. if (_end_benchmark_measure) {
  353. JNIEnv *env = get_jni_env();
  354. ERR_FAIL_NULL(env);
  355. jstring j_context = env->NewStringUTF(p_context.utf8().get_data());
  356. jstring j_label = env->NewStringUTF(p_label.utf8().get_data());
  357. env->CallVoidMethod(godot_instance, _end_benchmark_measure, j_context, j_label);
  358. env->DeleteLocalRef(j_context);
  359. env->DeleteLocalRef(j_label);
  360. }
  361. }
  362. void GodotJavaWrapper::dump_benchmark(const String &benchmark_file) {
  363. if (_dump_benchmark) {
  364. JNIEnv *env = get_jni_env();
  365. ERR_FAIL_NULL(env);
  366. jstring j_benchmark_file = env->NewStringUTF(benchmark_file.utf8().get_data());
  367. env->CallVoidMethod(godot_instance, _dump_benchmark, j_benchmark_file);
  368. env->DeleteLocalRef(j_benchmark_file);
  369. }
  370. }
  371. bool GodotJavaWrapper::has_feature(const String &p_feature) const {
  372. if (_has_feature) {
  373. JNIEnv *env = get_jni_env();
  374. ERR_FAIL_NULL_V(env, false);
  375. jstring j_feature = env->NewStringUTF(p_feature.utf8().get_data());
  376. bool result = env->CallBooleanMethod(godot_instance, _has_feature, j_feature);
  377. env->DeleteLocalRef(j_feature);
  378. return result;
  379. } else {
  380. return false;
  381. }
  382. }