T2DActivity.cpp 81 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2013 GarageGames, LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. #include "platformAndroid/T2DActivity.h"
  23. #include "platformAndroid/platformGL.h"
  24. #include "platformAndroid/AndroidWindow.h"
  25. #include "platformAndroid/platformAndroid.h"
  26. #include "graphics/dgl.h"
  27. #include "platform/event.h"
  28. #include "game/gameInterface.h"
  29. extern AndroidPlatState platState;
  30. static void engine_term_display(struct engine* engine, bool shutdown);
  31. extern void android_StartupTime();
  32. T2DActivity activity;
  33. bool keyboardShowing = false;
  34. float keyboardTransition = 1.0f;
  35. bool bSuspended = false;
  36. bool SetupCompleted = false;
  37. double lastSystemTime = 0;
  38. #define USE_DEPTH_BUFFER 0
  39. extern int _AndroidRunTorqueMain(engine *eng);
  40. extern bool createMouseMoveEvent(S32 i, S32 x, S32 y, S32 lastX, S32 lastY);
  41. extern bool createMouseDownEvent(S32 touchNumber, S32 x, S32 y, U32 numTouches);
  42. extern bool createMouseUpEvent(S32 touchNumber, S32 x, S32 y, S32 lastX, S32 lastY, U32 numTouches); //EFM
  43. extern void createMouseTapEvent( S32 nbrTaps, S32 x, S32 y );
  44. extern void _AndroidGameInnerLoop();
  45. extern void _AndroidGameResignActive();
  46. extern void _AndroidGameBecomeActive();
  47. extern void _AndroidGameWillTerminate();
  48. S32 _AndroidGameGetOrientation()
  49. {
  50. return AConfiguration_getOrientation(platState.engine->app->config);
  51. }
  52. bool _AndroidTorqueFatalError = false;
  53. extern void _AndroidGameInnerLoop();
  54. double timeGetTime() {
  55. struct timeval tv;
  56. gettimeofday(&tv, NULL);
  57. return ((tv.tv_sec) * 1000.0 + (tv.tv_usec) / 1000.0);
  58. }
  59. void toggleSplashScreen(bool show)
  60. {
  61. // Attaches the current thread to the JVM.
  62. jint lResult;
  63. jint lFlags = 0;
  64. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  65. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  66. JavaVMAttachArgs lJavaVMAttachArgs;
  67. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  68. lJavaVMAttachArgs.name = "NativeThread";
  69. lJavaVMAttachArgs.group = NULL;
  70. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  71. if (lResult == JNI_ERR) {
  72. return;
  73. }
  74. // Retrieves NativeActivity.
  75. jobject lNativeActivity = platState.engine->app->activity->clazz;
  76. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  77. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  78. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  79. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  80. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  81. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/SplashScreen");
  82. jclass SplashScreenClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  83. jmethodID MethodSplashScreen = lJNIEnv->GetStaticMethodID(SplashScreenClass, "ToggleSplashScreen", "(Landroid/content/Context;ZII)V");
  84. lJNIEnv->CallStaticVoidMethod(SplashScreenClass, MethodSplashScreen, lNativeActivity, (jboolean)show, (jint)platState.engine->width, (jint)platState.engine->height);
  85. // Finished with the JVM.
  86. lJavaVM->DetachCurrentThread();
  87. }
  88. void ChangeVolume(bool up) {
  89. // Attaches the current thread to the JVM.
  90. jint lResult;
  91. jint lFlags = 0;
  92. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  93. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  94. JavaVMAttachArgs lJavaVMAttachArgs;
  95. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  96. lJavaVMAttachArgs.name = "NativeThread";
  97. lJavaVMAttachArgs.group = NULL;
  98. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  99. if (lResult == JNI_ERR) {
  100. return;
  101. }
  102. // Retrieves NativeActivity.
  103. jobject lNativeActivity = platState.engine->app->activity->clazz;
  104. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  105. // Retrieves Context.AUDIO_SERVICE.
  106. jclass ClassContext = lJNIEnv->FindClass("android/content/Context");
  107. jfieldID FieldAUDIO_SERVICE =lJNIEnv->GetStaticFieldID(ClassContext,"AUDIO_SERVICE", "Ljava/lang/String;");
  108. jobject AUDIO_SERVICE = lJNIEnv->GetStaticObjectField(ClassContext, FieldAUDIO_SERVICE);
  109. // Runs getSystemService(AUDIO_SERVICE)
  110. jclass ClassAudioManager = lJNIEnv->FindClass("android/media/AudioManager");
  111. jmethodID MethodGetSystemService = lJNIEnv->GetMethodID(ClassNativeActivity, "getSystemService","(Ljava/lang/String;)Ljava/lang/Object;");
  112. jobject lAudioManager = lJNIEnv->CallObjectMethod(lNativeActivity, MethodGetSystemService, AUDIO_SERVICE);
  113. //get AudioManager.STREAM_MUSIC
  114. jfieldID FieldSTREAM_MUSIC =lJNIEnv->GetStaticFieldID(ClassAudioManager,"STREAM_MUSIC", "I");
  115. jint STREAM_MUSIC = lJNIEnv->GetStaticIntField(ClassAudioManager, FieldSTREAM_MUSIC);
  116. //get AudioManager.FLAG_SHOW_UI
  117. jfieldID FieldFLAG_SHOW_UI =lJNIEnv->GetStaticFieldID(ClassAudioManager,"FLAG_SHOW_UI", "I");
  118. jint FLAG_SHOW_UI = lJNIEnv->GetStaticIntField(ClassAudioManager, FieldFLAG_SHOW_UI);
  119. if (up) {
  120. //get AudioManager.ADJUST_RAISE
  121. jfieldID FieldADJUST_RAISE =lJNIEnv->GetStaticFieldID(ClassAudioManager,"ADJUST_RAISE", "I");
  122. jint ADJUST_RAISE = lJNIEnv->GetStaticIntField(ClassAudioManager, FieldADJUST_RAISE);
  123. //audio.adjustStreamVolume(AudioManager.STREAM_MUSIC, AudioManager.ADJUST_RAISE, AudioManager.FLAG_SHOW_UI);
  124. jmethodID MethodAdjustStreamVolume = lJNIEnv->GetMethodID(ClassAudioManager, "adjustStreamVolume", "(III)V");
  125. lJNIEnv->CallVoidMethod(lAudioManager,MethodAdjustStreamVolume, STREAM_MUSIC, ADJUST_RAISE, FLAG_SHOW_UI);
  126. } else {
  127. //get AudioManager.ADJUST_LOWER
  128. jfieldID FieldADJUST_LOWER =lJNIEnv->GetStaticFieldID(ClassAudioManager,"ADJUST_LOWER", "I");
  129. jint ADJUST_LOWER = lJNIEnv->GetStaticIntField(ClassAudioManager, FieldADJUST_LOWER);
  130. //audio.adjustStreamVolume(AudioManager.STREAM_MUSIC, AudioManager.ADJUST_LOWER, AudioManager.FLAG_SHOW_UI);
  131. jmethodID MethodAdjustStreamVolume = lJNIEnv->GetMethodID(ClassAudioManager, "adjustStreamVolume", "(III)V");
  132. lJNIEnv->CallVoidMethod(lAudioManager,MethodAdjustStreamVolume, STREAM_MUSIC, ADJUST_LOWER, FLAG_SHOW_UI);
  133. }
  134. // Finished with the JVM.
  135. lJavaVM->DetachCurrentThread();
  136. }
  137. int convertAndroidToWindowsKeyCode(int key) {
  138. switch(key) {
  139. case AKEYCODE_0:
  140. return 48;
  141. case AKEYCODE_1:
  142. return 49;
  143. case AKEYCODE_2:
  144. return 50;
  145. case AKEYCODE_3:
  146. return 51;
  147. case AKEYCODE_4:
  148. return 52;
  149. case AKEYCODE_5:
  150. return 53;
  151. case AKEYCODE_6:
  152. return 54;
  153. case AKEYCODE_7:
  154. return 55;
  155. case AKEYCODE_8:
  156. return 56;
  157. case AKEYCODE_9:
  158. return 57;
  159. case AKEYCODE_A:
  160. return 65;
  161. case AKEYCODE_ALT_LEFT:
  162. return 0xA4;
  163. case AKEYCODE_ALT_RIGHT:
  164. return 0xA5;
  165. case AKEYCODE_APOSTROPHE:
  166. return 39;
  167. case AKEYCODE_AT:
  168. return 64;
  169. case AKEYCODE_B:
  170. return 66;
  171. case AKEYCODE_BACKSLASH:
  172. return 92;
  173. case AKEYCODE_C:
  174. return 67;
  175. //case KEYCODE_CAPS_LOCK:
  176. // return 0x14;
  177. case AKEYCODE_COMMA:
  178. return 44;
  179. case AKEYCODE_D:
  180. return 68;
  181. case AKEYCODE_DEL:
  182. return 8;
  183. case AKEYCODE_E:
  184. return 69;
  185. case AKEYCODE_ENTER:
  186. return 13;
  187. case AKEYCODE_EQUALS:
  188. return 61;
  189. //case AKEYCODE_ESCAPE:
  190. // return 0x1B;
  191. case AKEYCODE_F:
  192. return 70;
  193. //case AKEYCODE_F1:
  194. // return 0x70;
  195. //case AKEYCODE_F10:
  196. // return 0x79;
  197. //case AKEYCODE_F11:
  198. // return 0x7A;
  199. //case AKEYCODE_F12:
  200. // return 0x7B;
  201. //case AKEYCODE_F2:
  202. // return 0x71;
  203. //case AKEYCODE_F3:
  204. // return 0x72;
  205. //case AKEYCODE_F4:
  206. // return 0x73;
  207. //case AKEYCODE_F5:
  208. // return 0x74;
  209. //case AKEYCODE_F6:
  210. // return 0x75;
  211. //case AKEYCODE_F7:
  212. // return 0x76;
  213. //case AKEYCODE_F8:
  214. // return 0x77;
  215. //case AKEYCODE_F9:
  216. // return 0x78;
  217. case AKEYCODE_G:
  218. return 71;
  219. case AKEYCODE_GRAVE:
  220. return 96;
  221. case AKEYCODE_H:
  222. return 72;
  223. case AKEYCODE_I:
  224. return 73;
  225. //case AKEYCODE_INSERT:
  226. // return 0x2D;
  227. case AKEYCODE_J:
  228. return 74;
  229. case AKEYCODE_K:
  230. return 75;
  231. case AKEYCODE_L:
  232. return 76;
  233. case AKEYCODE_LEFT_BRACKET:
  234. return 91;
  235. case AKEYCODE_M:
  236. return 77;
  237. case AKEYCODE_MINUS:
  238. return 45;
  239. case AKEYCODE_N:
  240. return 78;
  241. case AKEYCODE_O:
  242. return 79;
  243. case AKEYCODE_P:
  244. return 80;
  245. case AKEYCODE_PAGE_DOWN:
  246. return 0x22;
  247. case AKEYCODE_PAGE_UP:
  248. return 0x21;
  249. case AKEYCODE_PERIOD:
  250. return 46;
  251. case AKEYCODE_PLUS:
  252. return 43;
  253. case AKEYCODE_POUND:
  254. return 35;
  255. case AKEYCODE_Q:
  256. return 81;
  257. case AKEYCODE_R:
  258. return 82;
  259. case AKEYCODE_RIGHT_BRACKET:
  260. return 93;
  261. case AKEYCODE_S:
  262. return 83;
  263. case AKEYCODE_SEMICOLON:
  264. return 59;
  265. case AKEYCODE_SHIFT_LEFT:
  266. return 0xA0;
  267. case AKEYCODE_SHIFT_RIGHT:
  268. return 0xA1;
  269. case AKEYCODE_SLASH:
  270. return 47;
  271. case AKEYCODE_SPACE:
  272. return 32;
  273. case AKEYCODE_STAR:
  274. return 42;
  275. case AKEYCODE_T:
  276. return 84;
  277. case AKEYCODE_TAB:
  278. return 0x09;
  279. case AKEYCODE_U:
  280. return 85;
  281. case AKEYCODE_V:
  282. return 86;
  283. case AKEYCODE_W:
  284. return 87;
  285. case AKEYCODE_X:
  286. return 88;
  287. case AKEYCODE_Y:
  288. return 89;
  289. case AKEYCODE_Z:
  290. return 90;
  291. default:
  292. return 0;
  293. }
  294. return 0;
  295. }
  296. void androidKeyboardEvent(int keyval, bool make) {
  297. S32 keyCode = convertAndroidToWindowsKeyCode(keyval);
  298. InputEvent event;
  299. event.deviceInst = 0;
  300. event.deviceType = KeyboardDeviceType;
  301. event.objType = SI_KEY;
  302. event.objInst = keyCode;
  303. event.action = make ? SI_MAKE : SI_BREAK;
  304. event.modifier = 0;
  305. event.ascii = keyCode;
  306. event.fValue = make ? 1.0f : 0.0f;
  307. Game->postEvent(event);
  308. }
  309. Point2I rawLastTouches[10];
  310. // Handle touch and keyboard input from android OS
  311. static int32_t engine_handle_input(struct android_app* app, AInputEvent* event) {
  312. struct engine* engine = (struct engine*)app->userData;
  313. if (AInputEvent_getType(event) == AINPUT_EVENT_TYPE_MOTION) {
  314. if ((AMotionEvent_getAction(event) & AMOTION_EVENT_ACTION_MASK) == AMOTION_EVENT_ACTION_DOWN) {
  315. size_t touchCount = AMotionEvent_getPointerCount(event);
  316. for (U8 i = 0; i < touchCount; i++)
  317. {
  318. Point2I point;
  319. point.x = AMotionEvent_getX(event, i);
  320. point.y = AMotionEvent_getY(event, i);
  321. rawLastTouches[i].x = point.x;
  322. rawLastTouches[i].y = point.y;
  323. createMouseDownEvent(i, point.x, point.y, touchCount);
  324. }
  325. }
  326. if ((AMotionEvent_getAction(event) & AMOTION_EVENT_ACTION_MASK) == AMOTION_EVENT_ACTION_UP) {
  327. size_t touchCount = AMotionEvent_getPointerCount(event);
  328. for (U8 i = 0; i < touchCount; i++)
  329. {
  330. Point2I point;
  331. point.x = AMotionEvent_getX(event, i);
  332. point.y = AMotionEvent_getY(event, i);
  333. Point2I prevPoint = rawLastTouches[i];
  334. createMouseUpEvent(i, point.x, point.y, prevPoint.x, prevPoint.y, touchCount);
  335. if (touchCount > 0)
  336. {
  337. // this was a tap, so create a tap event
  338. createMouseTapEvent(touchCount, (int) point.x, (int) point.y);
  339. }
  340. }
  341. }
  342. if ((AMotionEvent_getAction(event) & AMOTION_EVENT_ACTION_MASK) == AMOTION_EVENT_ACTION_MOVE) {
  343. size_t touchCount = AMotionEvent_getPointerCount(event);
  344. for (U8 i = 0; i < touchCount; i++)
  345. {
  346. Point2I point;
  347. point.x = AMotionEvent_getX(event, i);
  348. point.y = AMotionEvent_getY(event, i);
  349. Point2I prevPoint = rawLastTouches[i];
  350. createMouseMoveEvent(i, point.x, point.y, prevPoint.x, prevPoint.y);
  351. rawLastTouches[i].x = point.x;
  352. rawLastTouches[i].y = point.y;
  353. }
  354. }
  355. if ((AMotionEvent_getAction(event) & AMOTION_EVENT_ACTION_MASK) == AMOTION_EVENT_ACTION_CANCEL) {
  356. size_t touchCount = AMotionEvent_getPointerCount(event);
  357. for (U8 i = 0; i < touchCount; i++)
  358. {
  359. Point2I point;
  360. point.x = AMotionEvent_getX(event, i);
  361. point.y = AMotionEvent_getY(event, i);
  362. Point2I prevPoint = rawLastTouches[i];
  363. createMouseUpEvent(i, point.x, point.y, prevPoint.x, prevPoint.y, touchCount);
  364. if (touchCount > 0)
  365. {
  366. // this was a tap, so create a tap event
  367. createMouseTapEvent(touchCount, (int) point.x, (int) point.y);
  368. }
  369. }
  370. }
  371. return 1;
  372. } else if (AInputEvent_getType(event) == AINPUT_EVENT_TYPE_KEY) {
  373. int action = AKeyEvent_getAction(event);
  374. int key_val = AKeyEvent_getKeyCode(event);
  375. int metastate = AKeyEvent_getMetaState(event);
  376. switch(action)
  377. {
  378. case AKEY_EVENT_ACTION_DOWN:
  379. if (key_val == AKEYCODE_VOLUME_UP) {
  380. ChangeVolume(true);
  381. } else if (key_val == AKEYCODE_VOLUME_DOWN) {
  382. ChangeVolume(false);
  383. } else if (key_val == AKEYCODE_BACK) {
  384. Con::executef(2, "androidBackButton", Con::getBoolArg(true));
  385. } else {
  386. androidKeyboardEvent(key_val, true);
  387. }
  388. break;
  389. case AKEY_EVENT_ACTION_UP:
  390. if (key_val == AKEYCODE_VOLUME_UP) {
  391. ChangeVolume(true);
  392. } else if (key_val == AKEYCODE_VOLUME_DOWN) {
  393. ChangeVolume(false);
  394. } else if (key_val == AKEYCODE_BACK) {
  395. Con::executef(2, "androidBackButton", Con::getBoolArg(false));
  396. } else {
  397. androidKeyboardEvent(key_val, false);
  398. }
  399. break;
  400. }
  401. return 1;
  402. }
  403. return 0;
  404. }
  405. int _AndroidGetScreenWidth() {
  406. return platState.engine->width;
  407. }
  408. int _AndroidGetScreenHeight() {
  409. return platState.engine->height;
  410. }
  411. bool _AndroidGetFileDescriptor(const char* fileName, int32_t *mDescriptor, off_t *mStart, off_t* mLength) {
  412. AAsset* asset = AAssetManager_open(platState.engine->app->activity->assetManager, fileName, AASSET_MODE_UNKNOWN);
  413. if (asset != NULL) {
  414. *mDescriptor = AAsset_openFileDescriptor(asset, mStart, mLength);
  415. AAsset_close(asset);
  416. return true;
  417. }
  418. *mDescriptor = 0;
  419. *mStart = 0;
  420. *mLength = 0;
  421. return false;
  422. }
  423. char* _AndroidLoadFile(const char* fn, U32 *size) {
  424. char fileName[255];
  425. if (fn[0] == '/')
  426. {
  427. strcpy(fileName, fn+1);
  428. }
  429. else
  430. {
  431. strcpy(fileName,fn);
  432. }
  433. AAsset *asset;
  434. uint8_t buf[1024];
  435. char* buffer = NULL;
  436. *size = 0;
  437. asset = AAssetManager_open(platState.engine->app->activity->assetManager, fileName, AASSET_MODE_UNKNOWN);
  438. if(asset != NULL){
  439. *size = AAsset_getLength(asset);
  440. if (*size <= 0)
  441. return NULL;
  442. buffer = new char[*size + 1];
  443. int count = 0;
  444. while(true)
  445. {
  446. int read = AAsset_read(asset, buf, 1024);
  447. if (read <= 0)
  448. break;
  449. memcpy(buffer+count,buf, read);
  450. count += read;
  451. }
  452. buffer[*size] = '\0';
  453. AAsset_close(asset);
  454. }
  455. return buffer;
  456. }
  457. void _AndroidGetDeviceIPAddress(char* address) {
  458. int fd;
  459. strcpy(address, "error");
  460. // Attaches the current thread to the JVM.
  461. jint lResult;
  462. jint lFlags = 0;
  463. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  464. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  465. JavaVMAttachArgs lJavaVMAttachArgs;
  466. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  467. lJavaVMAttachArgs.name = "NativeThread";
  468. lJavaVMAttachArgs.group = NULL;
  469. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  470. if (lResult == JNI_ERR) {
  471. return;
  472. }
  473. // Retrieves NativeActivity.
  474. jobject lNativeActivity = platState.engine->app->activity->clazz;
  475. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  476. // Retrieves Context.INPUT_METHOD_SERVICE.
  477. jclass ClassContext = lJNIEnv->FindClass("android/content/Context");
  478. jfieldID FieldWIFI_SERVICE =
  479. lJNIEnv->GetStaticFieldID(ClassContext,
  480. "WIFI_SERVICE", "Ljava/lang/String;");
  481. jobject WIFI_SERVICE =
  482. lJNIEnv->GetStaticObjectField(ClassContext,
  483. FieldWIFI_SERVICE);
  484. // Runs getSystemService(Context.WIFI_SERVICE).
  485. jclass ClassWifiManager = lJNIEnv->FindClass(
  486. "android/net/wifi/WifiManager");
  487. jclass ClassWifiInfo = lJNIEnv->FindClass(
  488. "android/net/wifi/WifiInfo");
  489. jmethodID MethodGetSystemService = lJNIEnv->GetMethodID(
  490. ClassNativeActivity, "getSystemService",
  491. "(Ljava/lang/String;)Ljava/lang/Object;");
  492. jobject lWifiManager = lJNIEnv->CallObjectMethod(
  493. lNativeActivity, MethodGetSystemService,
  494. WIFI_SERVICE);
  495. // Runs wifiManager.getConnectionInfo()
  496. jmethodID MethodGetConnectionInfo = lJNIEnv->GetMethodID(
  497. ClassWifiManager, "getConnectionInfo",
  498. "()Landroid/net/wifi/WifiInfo;");
  499. jobject lWifiInfo = lJNIEnv->CallObjectMethod(
  500. lWifiManager, MethodGetConnectionInfo);
  501. // Runs wifiInfo.getIpAddress()
  502. jmethodID MethodGetIPAddress = lJNIEnv->GetMethodID(
  503. ClassWifiInfo, "getIpAddress",
  504. "()I");
  505. jint lIPAddress = lJNIEnv->CallIntMethod(
  506. lWifiInfo, MethodGetIPAddress);
  507. int ip = lIPAddress;
  508. // Finished with the JVM.
  509. lJavaVM->DetachCurrentThread();
  510. char buffer[32];
  511. sprintf(buffer, "%d.%d.%d.%d", (ip & 0xFF), (ip >> 8 & 0xFF), (ip >> 16 & 0xFF), (ip >> 24 & 0xFF));
  512. strcpy(address, buffer);
  513. }
  514. /**
  515. * Initialize an EGL context for the current display.
  516. */
  517. static int engine_init_display(struct engine* engine) {
  518. // initialize OpenGL ES and EGL
  519. /*
  520. * Here specify the attributes of the desired configuration.
  521. * Below, we select an EGLConfig with at least 8 bits per color
  522. * component compatible with on-screen windows
  523. */
  524. const EGLint attribs[] = {
  525. EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
  526. EGL_BLUE_SIZE, 8,
  527. EGL_GREEN_SIZE, 8,
  528. EGL_RED_SIZE, 8,
  529. //EGL_ALPHA_SIZE, 8,
  530. //EGL_DEPTH_SIZE, 24,
  531. EGL_RENDERABLE_TYPE, EGL_OPENGL_ES_BIT,
  532. EGL_NONE
  533. };
  534. static const EGLint ctx_attribs[] = {
  535. EGL_CONTEXT_CLIENT_VERSION, 1,
  536. EGL_NONE
  537. };
  538. EGLint w, h, dummy, format;
  539. EGLint numConfigs;
  540. EGLConfig config;
  541. EGLSurface surface;
  542. EGLContext context;
  543. EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
  544. eglInitialize(display, 0, 0);
  545. /* Here, the application chooses the configuration it desires. In this
  546. * sample, we have a very simplified selection process, where we pick
  547. * the first EGLConfig that matches our criteria */
  548. eglChooseConfig(display, attribs, &config, 1, &numConfigs);
  549. /* EGL_NATIVE_VISUAL_ID is an attribute of the EGLConfig that is
  550. * guaranteed to be accepted by ANativeWindow_setBuffersGeometry().
  551. * As soon as we picked a EGLConfig, we can safely reconfigure the
  552. * ANativeWindow buffers to match, using EGL_NATIVE_VISUAL_ID. */
  553. eglGetConfigAttrib(display, config, EGL_NATIVE_VISUAL_ID, &format);
  554. ANativeWindow_setBuffersGeometry(engine->app->window, 0, 0, format);
  555. surface = eglCreateWindowSurface(display, config, engine->app->window, NULL);
  556. context = eglCreateContext(display, config, EGL_NO_CONTEXT, ctx_attribs);
  557. if (eglMakeCurrent(display, surface, surface, context) == EGL_FALSE) {
  558. adprintf("Unable to eglMakeCurrent");
  559. return -1;
  560. }
  561. eglQuerySurface(display, surface, EGL_WIDTH, &w);
  562. eglQuerySurface(display, surface, EGL_HEIGHT, &h);
  563. engine->display = display;
  564. engine->context = context;
  565. engine->surface = surface;
  566. engine->width = w;
  567. engine->height = h;
  568. engine->state.angle = 0;
  569. glDisable(GL_DEPTH_TEST);
  570. glDisable(GL_CULL_FACE);
  571. engine->animating = 1;
  572. glViewport(0, 0, engine->width, engine->height);
  573. if (SetupCompleted == false)
  574. {
  575. toggleSplashScreen(true);
  576. if (_AndroidRunTorqueMain(engine) == 0)
  577. {
  578. engine->app->destroyRequested = 1;
  579. engine_term_display(engine, true);
  580. return -1;
  581. }
  582. activity.finishGLSetup();
  583. SetupCompleted = true;
  584. }
  585. else
  586. {
  587. if(!_AndroidTorqueFatalError)
  588. _AndroidGameBecomeActive();
  589. }
  590. return 0;
  591. }
  592. void T2DActivity::finishGLSetup()
  593. {
  594. platState.multipleTouchesEnabled = true;
  595. _AndroidTorqueFatalError = false;
  596. }
  597. void T2DActivity::finishShutdown()
  598. {
  599. }
  600. void T2DActivity::enableAccelerometer(bool enable) {
  601. accelerometerActive = enable;
  602. if (enable) {
  603. if (platState.engine->accelerometerSensor != NULL) {
  604. ASensorEventQueue_enableSensor(platState.engine->sensorEventQueue,
  605. platState.engine->accelerometerSensor);
  606. // We'd like to get 30 events per second (in us).
  607. ASensorEventQueue_setEventRate(platState.engine->sensorEventQueue,
  608. platState.engine->accelerometerSensor, (1000L/30)*1000);
  609. }
  610. } else {
  611. if (platState.engine->accelerometerSensor != NULL) {
  612. ASensorEventQueue_disableSensor(platState.engine->sensorEventQueue,
  613. platState.engine->accelerometerSensor);
  614. }
  615. }
  616. }
  617. /**
  618. * update callback
  619. */
  620. static void engine_update_frame(struct engine* engine) {
  621. if (bSuspended == true)
  622. return;
  623. double thisSysTime = timeGetTime();
  624. float timeElapsed = (thisSysTime-lastSystemTime)/1000.0f;
  625. if (timeElapsed > 1.0f)
  626. timeElapsed = 1.0f; // clamp it
  627. lastSystemTime = thisSysTime;
  628. if (keyboardShowing) {
  629. if (keyboardTransition > 0.0f) {
  630. keyboardTransition -= timeElapsed * 2.0f;
  631. if (keyboardTransition < 0.0f)
  632. keyboardTransition = 0.0f;
  633. }
  634. } else {
  635. if (keyboardTransition < 1.0f) {
  636. keyboardTransition += timeElapsed * 2.0f;
  637. if (keyboardTransition > 1.0f)
  638. keyboardTransition = 1.0f;
  639. }
  640. }
  641. if (SetupCompleted == true)
  642. _AndroidGameInnerLoop();
  643. }
  644. /**
  645. * Tear down the EGL context currently associated with the display.
  646. */
  647. static void engine_term_display(struct engine* engine, bool shutdown) {
  648. if (shutdown == true) {
  649. _AndroidGameWillTerminate();
  650. } else {
  651. _AndroidGameResignActive();
  652. }
  653. activity.finishShutdown();
  654. if (engine->display != EGL_NO_DISPLAY) {
  655. eglMakeCurrent(engine->display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
  656. if (engine->context != EGL_NO_CONTEXT) {
  657. eglDestroyContext(engine->display, engine->context);
  658. }
  659. if (engine->surface != EGL_NO_SURFACE) {
  660. eglDestroySurface(engine->display, engine->surface);
  661. }
  662. eglTerminate(engine->display);
  663. }
  664. engine->animating = 0;
  665. engine->display = EGL_NO_DISPLAY;
  666. engine->context = EGL_NO_CONTEXT;
  667. engine->surface = EGL_NO_SURFACE;
  668. }
  669. void keepScreenOn() {
  670. // Attaches the current thread to the JVM.
  671. jint lResult;
  672. jint lFlags = 0;
  673. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  674. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  675. JavaVMAttachArgs lJavaVMAttachArgs;
  676. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  677. lJavaVMAttachArgs.name = "NativeThread";
  678. lJavaVMAttachArgs.group = NULL;
  679. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  680. if (lResult == JNI_ERR) {
  681. return;
  682. }
  683. // Retrieves NativeActivity.
  684. jobject lNativeActivity = platState.engine->app->activity->clazz;
  685. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  686. // Runs getWindow().getDecorView().
  687. jmethodID MethodGetWindow = lJNIEnv->GetMethodID(
  688. ClassNativeActivity, "getWindow",
  689. "()Landroid/view/Window;");
  690. jobject lWindow = lJNIEnv->CallObjectMethod(lNativeActivity,
  691. MethodGetWindow);
  692. jclass ClassWindow = lJNIEnv->FindClass(
  693. "android/view/Window");
  694. jmethodID MethodGetDecorView = lJNIEnv->GetMethodID(
  695. ClassWindow, "getDecorView", "()Landroid/view/View;");
  696. jobject lDecorView = lJNIEnv->CallObjectMethod(lWindow,
  697. MethodGetDecorView);
  698. jclass ClassView = lJNIEnv->FindClass("android/view/View");
  699. jmethodID MethodSetKeepScreenOn = lJNIEnv->GetMethodID(
  700. ClassView, "setKeepScreenOn", "(Z)V");
  701. jboolean on = true;
  702. lJNIEnv->CallVoidMethod(lDecorView, MethodSetKeepScreenOn, on);
  703. // Finished with the JVM.
  704. lJavaVM->DetachCurrentThread();
  705. }
  706. void T2DActivity::loadCacheDir() {
  707. // Attaches the current thread to the JVM.
  708. jint lResult;
  709. jint lFlags = 0;
  710. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  711. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  712. JavaVMAttachArgs lJavaVMAttachArgs;
  713. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  714. lJavaVMAttachArgs.name = "NativeThread";
  715. lJavaVMAttachArgs.group = NULL;
  716. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  717. if (lResult == JNI_ERR) {
  718. return;
  719. }
  720. // Retrieves NativeActivity.
  721. jobject lNativeActivity = platState.engine->app->activity->clazz;
  722. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  723. jmethodID getCacheDir = lJNIEnv->GetMethodID(ClassNativeActivity, "getCacheDir", "()Ljava/io/File;");
  724. jobject file = lJNIEnv->CallObjectMethod(platState.engine->app->activity->clazz, getCacheDir);
  725. jclass fileClass = lJNIEnv->FindClass("java/io/File");
  726. jmethodID getAbsolutePath = lJNIEnv->GetMethodID(fileClass, "getAbsolutePath", "()Ljava/lang/String;");
  727. jstring jpath = (jstring)lJNIEnv->CallObjectMethod(file, getAbsolutePath);
  728. const char* app_dir = lJNIEnv->GetStringUTFChars(jpath, NULL);
  729. strcpy(cacheDir, app_dir);
  730. cacheDir[strlen(app_dir)] = '\0';
  731. lJNIEnv->ReleaseStringUTFChars(jpath, app_dir);
  732. // Finished with the JVM.
  733. lJavaVM->DetachCurrentThread();
  734. }
  735. void T2DActivity::enumerateFonts() {
  736. // Attaches the current thread to the JVM.
  737. jint lResult;
  738. jint lFlags = 0;
  739. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  740. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  741. JavaVMAttachArgs lJavaVMAttachArgs;
  742. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  743. lJavaVMAttachArgs.name = "NativeThread";
  744. lJavaVMAttachArgs.group = NULL;
  745. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  746. if (lResult == JNI_ERR) {
  747. return;
  748. }
  749. // Retrieves NativeActivity.
  750. jobject lNativeActivity = platState.engine->app->activity->clazz;
  751. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  752. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  753. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  754. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  755. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  756. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FontManager");
  757. jclass FontManagerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  758. jmethodID MethodFontManager = lJNIEnv->GetStaticMethodID(FontManagerClass, "enumerateFonts", "(Landroid/content/Context;)V");
  759. lJNIEnv->CallStaticObjectMethod(FontManagerClass, MethodFontManager, lNativeActivity);
  760. lJNIEnv->DeleteLocalRef(strClassName);
  761. // Finished with the JVM.
  762. lJavaVM->DetachCurrentThread();
  763. }
  764. void T2DActivity::dumpFontList() {
  765. // Attaches the current thread to the JVM.
  766. jint lResult;
  767. jint lFlags = 0;
  768. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  769. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  770. JavaVMAttachArgs lJavaVMAttachArgs;
  771. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  772. lJavaVMAttachArgs.name = "NativeThread";
  773. lJavaVMAttachArgs.group = NULL;
  774. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  775. if (lResult == JNI_ERR) {
  776. return;
  777. }
  778. // Retrieves NativeActivity.
  779. jobject lNativeActivity = platState.engine->app->activity->clazz;
  780. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  781. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  782. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  783. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  784. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  785. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FontManager");
  786. jclass FontManagerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  787. jmethodID MethodFontManager = lJNIEnv->GetStaticMethodID(FontManagerClass, "dumpFontList", "()V");
  788. lJNIEnv->CallStaticObjectMethod(FontManagerClass, MethodFontManager);
  789. lJNIEnv->DeleteLocalRef(strClassName);
  790. // Finished with the JVM.
  791. lJavaVM->DetachCurrentThread();
  792. }
  793. void T2DActivity::getFontPath(const char* fontName, char* fontPath) {
  794. // Attaches the current thread to the JVM.
  795. jint lResult;
  796. jint lFlags = 0;
  797. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  798. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  799. JavaVMAttachArgs lJavaVMAttachArgs;
  800. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  801. lJavaVMAttachArgs.name = "NativeThread";
  802. lJavaVMAttachArgs.group = NULL;
  803. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  804. if (lResult == JNI_ERR) {
  805. return;
  806. }
  807. // Retrieves NativeActivity.
  808. jobject lNativeActivity = platState.engine->app->activity->clazz;
  809. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  810. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  811. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  812. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  813. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  814. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FontManager");
  815. jstring strFontName = lJNIEnv->NewStringUTF(fontName);
  816. jclass FontManagerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  817. jmethodID MethodFontManager = lJNIEnv->GetStaticMethodID(FontManagerClass, "getFont", "(Ljava/lang/String;)Ljava/lang/String;");
  818. jstring jpath = (jstring)lJNIEnv->CallStaticObjectMethod(FontManagerClass, MethodFontManager, strFontName);
  819. if (jpath != NULL)
  820. {
  821. const char* path = lJNIEnv->GetStringUTFChars(jpath, NULL);
  822. strcpy(fontPath, path);
  823. fontPath[strlen(path)] = '\0';
  824. lJNIEnv->ReleaseStringUTFChars(jpath, path);
  825. }
  826. else
  827. {
  828. strcpy(fontPath,"");
  829. }
  830. lJNIEnv->DeleteLocalRef(strClassName);
  831. lJNIEnv->DeleteLocalRef(strFontName);
  832. // Finished with the JVM.
  833. lJavaVM->DetachCurrentThread();
  834. }
  835. /**
  836. * Process the next main command.
  837. */
  838. static void engine_handle_cmd(struct android_app* app, int32_t cmd) {
  839. struct engine* engine = (struct engine*)app->userData;
  840. switch (cmd) {
  841. case APP_CMD_SAVE_STATE:
  842. // The system has asked us to save our current state. Do so.
  843. engine->app->savedState = malloc(sizeof(struct saved_state));
  844. *((struct saved_state*)engine->app->savedState) = engine->state;
  845. engine->app->savedStateSize = sizeof(struct saved_state);
  846. break;
  847. case APP_CMD_INIT_WINDOW:
  848. // The window is being shown, get it ready.
  849. if (engine->app->window != NULL) {
  850. engine_init_display(engine);
  851. if (bSuspended == true) {
  852. glViewport(0, 0, engine->width, engine->height);
  853. Game->textureResurrect();
  854. bSuspended = false;
  855. }
  856. }
  857. break;
  858. case APP_CMD_TERM_WINDOW:
  859. bSuspended = true;
  860. Game->textureKill();
  861. // The window is being hidden or closed, clean it up.
  862. engine_term_display(engine, false);
  863. break;
  864. case APP_CMD_RESUME:
  865. break;
  866. case APP_CMD_GAINED_FOCUS:
  867. // When our app gains focus, we start monitoring the accelerometer.
  868. if (engine->accelerometerSensor != NULL && activity.isAccelerometerActive()) {
  869. ASensorEventQueue_enableSensor(engine->sensorEventQueue,
  870. engine->accelerometerSensor);
  871. // We'd like to get 30 events per second (in us).
  872. ASensorEventQueue_setEventRate(engine->sensorEventQueue,
  873. engine->accelerometerSensor, (1000L/30)*1000);
  874. }
  875. engine->animating = 1;
  876. break;
  877. case APP_CMD_LOST_FOCUS:
  878. // When our app loses focus, we stop monitoring the accelerometer.
  879. // This is to avoid consuming battery while not being used.
  880. if (engine->accelerometerSensor != NULL) {
  881. ASensorEventQueue_disableSensor(engine->sensorEventQueue,
  882. engine->accelerometerSensor);
  883. }
  884. // stop animating.
  885. engine->animating = 0;
  886. break;
  887. case APP_CMD_PAUSE:
  888. break;
  889. }
  890. }
  891. struct engine engine;
  892. /**
  893. * This is the main entry point of a native application that is using
  894. * android_native_app_glue. It runs in its own thread, with its own
  895. * event loop for receiving input events and doing other things.
  896. */
  897. void android_main(struct android_app* state) {
  898. //init startup time so U32 doesnt overflow
  899. android_StartupTime();
  900. // Make sure glue isn't stripped.
  901. app_dummy();
  902. platState.engine = &engine;
  903. memset(&engine, 0, sizeof(engine));
  904. state->userData = &engine;
  905. state->onAppCmd = engine_handle_cmd;
  906. state->onInputEvent = engine_handle_input;
  907. engine.app = state;
  908. // Prepare to monitor accelerometer
  909. engine.sensorManager = ASensorManager_getInstance();
  910. engine.accelerometerSensor = ASensorManager_getDefaultSensor(engine.sensorManager,
  911. ASENSOR_TYPE_ACCELEROMETER);
  912. engine.sensorEventQueue = ASensorManager_createEventQueue(engine.sensorManager,
  913. state->looper, LOOPER_ID_USER, NULL, NULL);
  914. if (state->savedState != NULL) {
  915. // We are starting with a previous saved state; restore from it.
  916. engine.state = *(struct saved_state*)state->savedState;
  917. }
  918. keepScreenOn();
  919. //This is to help the debugger catch up. If you dont have this you cant debug this early in the execution
  920. //so only uncomment when needed as it adds 10 seconds to startup
  921. //sleep(10);
  922. //store the cache dir
  923. activity.loadCacheDir();
  924. //enumerate fonts
  925. activity.enumerateFonts();
  926. platState.argc = 0;
  927. // loop waiting for stuff to do.
  928. while (1) {
  929. // Read all pending events.
  930. int ident;
  931. int events;
  932. struct android_poll_source* source;
  933. // If not animating, we will block forever waiting for events.
  934. // If animating, we loop until all events are read, then continue
  935. // to draw the next frame of animation.
  936. while ((ident=ALooper_pollAll(engine.animating ? 0 : -1, NULL, &events,
  937. (void**)&source)) >= 0) {
  938. // Process this event.
  939. if (source != NULL) {
  940. source->process(state, source);
  941. }
  942. // If a sensor has data, process it now.
  943. if (ident == LOOPER_ID_USER) {
  944. if (engine.accelerometerSensor != NULL && activity.isAccelerometerActive()) {
  945. ASensorEvent event;
  946. int i = 0;
  947. while (ASensorEventQueue_getEvents(engine.sensorEventQueue,
  948. &event, 1) > 0) {
  949. InputEvent inputEvent;
  950. U32 accelAxes[6] = { SI_ACCELX, SI_ACCELY, SI_ACCELZ, SI_GRAVX, SI_GRAVY, SI_GRAVZ };
  951. double userAcc[6] = { event.acceleration.x, event.acceleration.y, event.acceleration.z,0,0,0};
  952. inputEvent.deviceInst = 0;
  953. inputEvent.fValue = userAcc[i];
  954. inputEvent.deviceType = AccelerometerDeviceType;
  955. inputEvent.objType = accelAxes[i];
  956. inputEvent.objInst = i;
  957. inputEvent.action = SI_MOTION;
  958. inputEvent.modifier = 0;
  959. Game->postEvent(inputEvent);
  960. i++;
  961. }
  962. }
  963. }
  964. // Check if we are exiting.
  965. if (state->destroyRequested != 0) {
  966. engine_term_display(&engine, true);
  967. return;
  968. }
  969. }
  970. engine_update_frame(&engine);
  971. /*if (engine.animating) {
  972. // Drawing is throttled to the screen update rate, so there
  973. // is no need to do timing here.
  974. engine_draw_frame(&engine);
  975. }*/
  976. }
  977. engine_term_display(&engine, true);
  978. }
  979. struct MatchPathSeparator
  980. {
  981. bool operator()( char ch ) const
  982. {
  983. return ch == '/';
  984. }
  985. };
  986. bool android_DumpDirectoriesExtra(Vector<StringTableEntry> &directoryVector)
  987. {
  988. // Attaches the current thread to the JVM.
  989. jint lResult;
  990. jint lFlags = 0;
  991. JavaVM* lJavaVM = engine.app->activity->vm;
  992. JNIEnv* lJNIEnv = engine.app->activity->env;
  993. JavaVMAttachArgs lJavaVMAttachArgs;
  994. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  995. lJavaVMAttachArgs.name = "NativeThread";
  996. lJavaVMAttachArgs.group = NULL;
  997. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  998. if (lResult == JNI_ERR) {
  999. return false;
  1000. }
  1001. // Retrieves NativeActivity.
  1002. jobject lNativeActivity = engine.app->activity->clazz;
  1003. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1004. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1005. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1006. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1007. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1008. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1009. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1010. jmethodID MethodExtraPaths = lJNIEnv->GetStaticMethodID(FileWalkerClass, "getRestOfDump", "()[Ljava/lang/String;");
  1011. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodExtraPaths);
  1012. bool ret = true;
  1013. if (stringArray)
  1014. {
  1015. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1016. if (stringCount < 500)
  1017. ret = false;
  1018. for (int i=0; i<stringCount; i++) {
  1019. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1020. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1021. char str[255];
  1022. strcpy(str, rawString);
  1023. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1024. if (!Platform::isExcludedDirectory(str))
  1025. {
  1026. directoryVector.push_back(StringTable->insert(str));
  1027. }
  1028. }
  1029. lJNIEnv->DeleteLocalRef(stringArray);
  1030. }
  1031. else
  1032. {
  1033. ret = false;
  1034. }
  1035. lJNIEnv->DeleteLocalRef(strClassName);
  1036. // Finished with the JVM.
  1037. lJavaVM->DetachCurrentThread();
  1038. return ret;
  1039. }
  1040. static Vector<Platform::FileInfo> dumpPathBackup;
  1041. bool android_DumpDirectories(const char *basePath, const char *path, Vector<StringTableEntry> &directoryVector, S32 depth, bool noBasePath)
  1042. {
  1043. // Attaches the current thread to the JVM.
  1044. jint lResult;
  1045. jint lFlags = 0;
  1046. JavaVM* lJavaVM = engine.app->activity->vm;
  1047. JNIEnv* lJNIEnv = engine.app->activity->env;
  1048. JavaVMAttachArgs lJavaVMAttachArgs;
  1049. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1050. lJavaVMAttachArgs.name = "NativeThread";
  1051. lJavaVMAttachArgs.group = NULL;
  1052. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1053. if (lResult == JNI_ERR) {
  1054. return false;
  1055. }
  1056. // Retrieves NativeActivity.
  1057. jobject lNativeActivity = engine.app->activity->clazz;
  1058. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1059. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1060. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1061. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1062. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1063. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1064. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1065. jstring strDirName = lJNIEnv->NewStringUTF(basePath);
  1066. jstring strDirName2 = lJNIEnv->NewStringUTF(path);
  1067. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "DumpDirectories", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;ZZ)[Ljava/lang/String;");
  1068. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName, strDirName2, depth != 0, noBasePath);
  1069. bool ret = false;
  1070. if (stringArray)
  1071. {
  1072. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1073. for (int i=0; i<stringCount; i++) {
  1074. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1075. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1076. char str[255];
  1077. strcpy(str, rawString);
  1078. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1079. if (!Platform::isExcludedDirectory(str))
  1080. {
  1081. directoryVector.push_back(StringTable->insert(str));
  1082. }
  1083. }
  1084. lJNIEnv->DeleteLocalRef(stringArray);
  1085. bool keepGoing = false;
  1086. if (stringCount == 500)
  1087. {
  1088. keepGoing = true;
  1089. }
  1090. lJNIEnv->DeleteLocalRef(strClassName);
  1091. lJNIEnv->DeleteLocalRef(strDirName);
  1092. lJNIEnv->DeleteLocalRef(strDirName2);
  1093. // Finished with the JVM.
  1094. lJavaVM->DetachCurrentThread();
  1095. while (keepGoing == true)
  1096. keepGoing = android_DumpDirectoriesExtra(directoryVector);
  1097. ret = true;
  1098. }
  1099. else
  1100. {
  1101. lJNIEnv->DeleteLocalRef(strClassName);
  1102. lJNIEnv->DeleteLocalRef(strDirName);
  1103. lJNIEnv->DeleteLocalRef(strDirName2);
  1104. // Finished with the JVM.
  1105. lJavaVM->DetachCurrentThread();
  1106. ret = false;
  1107. }
  1108. return ret;
  1109. }
  1110. bool android_DumpPathExtra(Vector<Platform::FileInfo>& fileVector)
  1111. {
  1112. // Attaches the current thread to the JVM.
  1113. jint lResult;
  1114. jint lFlags = 0;
  1115. JavaVM* lJavaVM = engine.app->activity->vm;
  1116. JNIEnv* lJNIEnv = engine.app->activity->env;
  1117. JavaVMAttachArgs lJavaVMAttachArgs;
  1118. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1119. lJavaVMAttachArgs.name = "NativeThread";
  1120. lJavaVMAttachArgs.group = NULL;
  1121. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1122. if (lResult == JNI_ERR) {
  1123. return false;
  1124. }
  1125. // Retrieves NativeActivity.
  1126. jobject lNativeActivity = engine.app->activity->clazz;
  1127. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1128. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1129. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1130. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1131. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1132. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1133. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1134. jmethodID MethodExtraPaths = lJNIEnv->GetStaticMethodID(FileWalkerClass, "getRestOfDump", "()[Ljava/lang/String;");
  1135. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodExtraPaths);
  1136. bool ret = true;
  1137. if (stringArray)
  1138. {
  1139. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1140. if (stringCount < 500)
  1141. ret = false;
  1142. for (int i=0; i<stringCount; i++) {
  1143. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1144. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1145. std::string str = rawString;
  1146. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1147. const U32 fileSize = Platform::getFileSize(str.c_str());
  1148. fileVector.increment();
  1149. Platform::FileInfo& rInfo = fileVector.last();
  1150. std::string fileName = std::string(
  1151. std::find_if( str.rbegin(), str.rend(),
  1152. MatchPathSeparator() ).base(),
  1153. str.end() );
  1154. rInfo.pFullPath = StringTable->insert(str.substr(0,str.find(fileName)-1).c_str());
  1155. rInfo.pFileName = StringTable->insert(fileName.c_str());
  1156. rInfo.fileSize = fileSize;
  1157. dumpPathBackup.increment();
  1158. Platform::FileInfo& rInfo2 = dumpPathBackup.last();
  1159. rInfo2.pFullPath = rInfo.pFullPath;
  1160. rInfo2.pFileName = rInfo.pFileName;
  1161. rInfo2.fileSize = rInfo.fileSize;
  1162. }
  1163. lJNIEnv->DeleteLocalRef(stringArray);
  1164. }
  1165. else
  1166. {
  1167. ret = false;
  1168. }
  1169. lJNIEnv->DeleteLocalRef(strClassName);
  1170. // Finished with the JVM.
  1171. lJavaVM->DetachCurrentThread();
  1172. return ret;
  1173. }
  1174. bool loadDumpPathFromCache(const char* dir, Vector<Platform::FileInfo>& fileVector, U32 depth)
  1175. {
  1176. bool foundPaths = false;
  1177. for (int i = 0; i < dumpPathBackup.size(); i++)
  1178. {
  1179. if (depth == 0)
  1180. {
  1181. Platform::FileInfo fi = dumpPathBackup[i];
  1182. if (strcmp(fi.pFullPath, dir) == 0)
  1183. {
  1184. fileVector.increment();
  1185. Platform::FileInfo& rInfo = fileVector.last();
  1186. rInfo.pFileName = fi.pFileName;
  1187. rInfo.pFullPath = fi.pFullPath;
  1188. rInfo.fileSize = fi.fileSize;
  1189. foundPaths = true;
  1190. }
  1191. }
  1192. else
  1193. {
  1194. Platform::FileInfo fi = dumpPathBackup[i];
  1195. std::string fullPath = fi.pFullPath;
  1196. if (fullPath.find(dir) == 0)
  1197. {
  1198. fileVector.increment();
  1199. Platform::FileInfo& rInfo = fileVector.last();
  1200. rInfo.pFileName = fi.pFileName;
  1201. rInfo.pFullPath = fi.pFullPath;
  1202. rInfo.fileSize = fi.fileSize;
  1203. foundPaths = true;
  1204. }
  1205. }
  1206. }
  1207. return true;
  1208. }
  1209. bool android_DumpPath(const char* dir, Vector<Platform::FileInfo>& fileVector, U32 depth)
  1210. {
  1211. if (dumpPathBackup.size() > 0)
  1212. {
  1213. return loadDumpPathFromCache(dir, fileVector, depth);
  1214. }
  1215. // Attaches the current thread to the JVM.
  1216. jint lResult;
  1217. jint lFlags = 0;
  1218. JavaVM* lJavaVM = engine.app->activity->vm;
  1219. JNIEnv* lJNIEnv = engine.app->activity->env;
  1220. JavaVMAttachArgs lJavaVMAttachArgs;
  1221. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1222. lJavaVMAttachArgs.name = "NativeThread";
  1223. lJavaVMAttachArgs.group = NULL;
  1224. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1225. if (lResult == JNI_ERR) {
  1226. return false;
  1227. }
  1228. // Retrieves NativeActivity.
  1229. jobject lNativeActivity = engine.app->activity->clazz;
  1230. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1231. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1232. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1233. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1234. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1235. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1236. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1237. jstring strDirName = lJNIEnv->NewStringUTF(dir);
  1238. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "DumpPath", "(Landroid/content/Context;Ljava/lang/String;Z)[Ljava/lang/String;");
  1239. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName, depth != 0);
  1240. bool ret = false;
  1241. if (stringArray)
  1242. {
  1243. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1244. for (int i=0; i<stringCount; i++) {
  1245. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1246. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1247. std::string str = rawString;
  1248. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1249. const U32 fileSize = Platform::getFileSize(str.c_str());
  1250. fileVector.increment();
  1251. Platform::FileInfo& rInfo = fileVector.last();
  1252. std::string fileName = std::string(
  1253. std::find_if( str.rbegin(), str.rend(),
  1254. MatchPathSeparator() ).base(),
  1255. str.end() );
  1256. rInfo.pFullPath = StringTable->insert(str.substr(0,str.find(fileName)-1).c_str());
  1257. rInfo.pFileName = StringTable->insert(fileName.c_str());
  1258. rInfo.fileSize = fileSize;
  1259. dumpPathBackup.increment();
  1260. Platform::FileInfo& rInfo2 = dumpPathBackup.last();
  1261. rInfo2.pFullPath = rInfo.pFullPath;
  1262. rInfo2.pFileName = rInfo.pFileName;
  1263. rInfo2.fileSize = rInfo.fileSize;
  1264. }
  1265. lJNIEnv->DeleteLocalRef(stringArray);
  1266. stringArray = NULL;
  1267. lJNIEnv->DeleteLocalRef(strClassName);
  1268. lJNIEnv->DeleteLocalRef(strDirName);
  1269. // Finished with the JVM.
  1270. lJavaVM->DetachCurrentThread();
  1271. bool keepGoing = false;
  1272. if (stringCount == 500)
  1273. {
  1274. keepGoing = true;
  1275. }
  1276. while (keepGoing == true)
  1277. keepGoing = android_DumpPathExtra(fileVector);
  1278. ret = true;
  1279. }
  1280. else
  1281. {
  1282. lJNIEnv->DeleteLocalRef(strClassName);
  1283. lJNIEnv->DeleteLocalRef(strDirName);
  1284. // Finished with the JVM.
  1285. lJavaVM->DetachCurrentThread();
  1286. ret = false;
  1287. }
  1288. return ret;
  1289. }
  1290. void android_InitDirList(const char* dir)
  1291. {
  1292. // Attaches the current thread to the JVM.
  1293. jint lResult;
  1294. jint lFlags = 0;
  1295. JavaVM* lJavaVM = engine.app->activity->vm;
  1296. JNIEnv* lJNIEnv = engine.app->activity->env;
  1297. JavaVMAttachArgs lJavaVMAttachArgs;
  1298. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1299. lJavaVMAttachArgs.name = "NativeThread";
  1300. lJavaVMAttachArgs.group = NULL;
  1301. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1302. if (lResult == JNI_ERR) {
  1303. return;
  1304. }
  1305. // Retrieves NativeActivity.
  1306. jobject lNativeActivity = engine.app->activity->clazz;
  1307. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1308. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1309. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1310. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1311. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1312. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1313. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1314. jstring strDirName = lJNIEnv->NewStringUTF(dir);
  1315. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "InitDirList", "(Landroid/content/Context;Ljava/lang/String;)V");
  1316. lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName);
  1317. lJNIEnv->DeleteLocalRef(strClassName);
  1318. lJNIEnv->DeleteLocalRef(strDirName);
  1319. // Finished with the JVM.
  1320. lJavaVM->DetachCurrentThread();
  1321. }
  1322. void android_GetNextDir(const char* pdir, char *dir)
  1323. {
  1324. // Attaches the current thread to the JVM.
  1325. jint lResult;
  1326. jint lFlags = 0;
  1327. JavaVM* lJavaVM = engine.app->activity->vm;
  1328. JNIEnv* lJNIEnv = engine.app->activity->env;
  1329. JavaVMAttachArgs lJavaVMAttachArgs;
  1330. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1331. lJavaVMAttachArgs.name = "NativeThread";
  1332. lJavaVMAttachArgs.group = NULL;
  1333. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1334. if (lResult == JNI_ERR) {
  1335. return;
  1336. }
  1337. // Retrieves NativeActivity.
  1338. jobject lNativeActivity = engine.app->activity->clazz;
  1339. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1340. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1341. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1342. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1343. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1344. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1345. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1346. jstring strDirName = lJNIEnv->NewStringUTF(pdir);
  1347. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "GetNextDir", "(Ljava/lang/String;)Ljava/lang/String;");
  1348. jstring jpath = (jstring)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, strDirName);
  1349. if (jpath != NULL)
  1350. {
  1351. const char* path = lJNIEnv->GetStringUTFChars(jpath, NULL);
  1352. strcpy(dir, path);
  1353. dir[strlen(path)] = '\0';
  1354. lJNIEnv->ReleaseStringUTFChars(jpath, path);
  1355. }
  1356. else
  1357. {
  1358. strcpy(dir,"");
  1359. }
  1360. lJNIEnv->DeleteLocalRef(strClassName);
  1361. lJNIEnv->DeleteLocalRef(strDirName);
  1362. // Finished with the JVM.
  1363. lJavaVM->DetachCurrentThread();
  1364. }
  1365. void android_GetNextFile(const char* pdir, char *file)
  1366. {
  1367. // Attaches the current thread to the JVM.
  1368. jint lResult;
  1369. jint lFlags = 0;
  1370. JavaVM* lJavaVM = engine.app->activity->vm;
  1371. JNIEnv* lJNIEnv = engine.app->activity->env;
  1372. JavaVMAttachArgs lJavaVMAttachArgs;
  1373. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1374. lJavaVMAttachArgs.name = "NativeThread";
  1375. lJavaVMAttachArgs.group = NULL;
  1376. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1377. if (lResult == JNI_ERR) {
  1378. return;
  1379. }
  1380. // Retrieves NativeActivity.
  1381. jobject lNativeActivity = engine.app->activity->clazz;
  1382. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1383. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1384. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1385. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1386. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1387. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1388. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1389. jstring strDirName = lJNIEnv->NewStringUTF(pdir);
  1390. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "GetNextFile", "(Ljava/lang/String;)Ljava/lang/String;");
  1391. jstring jpath = (jstring)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, strDirName);
  1392. if (jpath != NULL)
  1393. {
  1394. const char* path = lJNIEnv->GetStringUTFChars(jpath, NULL);
  1395. strcpy(file, path);
  1396. file[strlen(path)] = '\0';
  1397. lJNIEnv->ReleaseStringUTFChars(jpath, path);
  1398. }
  1399. else
  1400. {
  1401. strcpy(file,"");
  1402. }
  1403. lJNIEnv->DeleteLocalRef(strClassName);
  1404. lJNIEnv->DeleteLocalRef(strDirName);
  1405. // Finished with the JVM.
  1406. lJavaVM->DetachCurrentThread();
  1407. }
  1408. bool android_IsFile(const char* path)
  1409. {
  1410. // Attaches the current thread to the JVM.
  1411. jint lResult;
  1412. jint lFlags = 0;
  1413. JavaVM* lJavaVM = engine.app->activity->vm;
  1414. JNIEnv* lJNIEnv = engine.app->activity->env;
  1415. JavaVMAttachArgs lJavaVMAttachArgs;
  1416. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1417. lJavaVMAttachArgs.name = "NativeThread";
  1418. lJavaVMAttachArgs.group = NULL;
  1419. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1420. if (lResult == JNI_ERR) {
  1421. return false;
  1422. }
  1423. // Retrieves NativeActivity.
  1424. jobject lNativeActivity = engine.app->activity->clazz;
  1425. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1426. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1427. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1428. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1429. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1430. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1431. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1432. jstring strDirName = lJNIEnv->NewStringUTF(path);
  1433. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "IsFile", "(Landroid/content/Context;Ljava/lang/String;)Z");
  1434. jboolean jpath = lJNIEnv->CallStaticBooleanMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName);
  1435. if (jpath)
  1436. {
  1437. lJNIEnv->DeleteLocalRef(strClassName);
  1438. lJNIEnv->DeleteLocalRef(strDirName);
  1439. // Finished with the JVM.
  1440. lJavaVM->DetachCurrentThread();
  1441. return true;
  1442. }
  1443. else
  1444. {
  1445. lJNIEnv->DeleteLocalRef(strClassName);
  1446. lJNIEnv->DeleteLocalRef(strDirName);
  1447. // Finished with the JVM.
  1448. lJavaVM->DetachCurrentThread();
  1449. return false;
  1450. }
  1451. }
  1452. bool android_IsDir(const char* path)
  1453. {
  1454. // Attaches the current thread to the JVM.
  1455. jint lResult;
  1456. jint lFlags = 0;
  1457. JavaVM* lJavaVM = engine.app->activity->vm;
  1458. JNIEnv* lJNIEnv = engine.app->activity->env;
  1459. JavaVMAttachArgs lJavaVMAttachArgs;
  1460. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1461. lJavaVMAttachArgs.name = "NativeThread";
  1462. lJavaVMAttachArgs.group = NULL;
  1463. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1464. if (lResult == JNI_ERR) {
  1465. return false;
  1466. }
  1467. // Retrieves NativeActivity.
  1468. jobject lNativeActivity = engine.app->activity->clazz;
  1469. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1470. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1471. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1472. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1473. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1474. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1475. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1476. jstring strDirName = lJNIEnv->NewStringUTF(path);
  1477. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "IsDir", "(Landroid/content/Context;Ljava/lang/String;)Z");
  1478. jboolean jpath = lJNIEnv->CallStaticBooleanMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName);
  1479. if (jpath)
  1480. {
  1481. lJNIEnv->DeleteLocalRef(strClassName);
  1482. lJNIEnv->DeleteLocalRef(strDirName);
  1483. // Finished with the JVM.
  1484. lJavaVM->DetachCurrentThread();
  1485. return true;
  1486. }
  1487. else
  1488. {
  1489. lJNIEnv->DeleteLocalRef(strClassName);
  1490. lJNIEnv->DeleteLocalRef(strDirName);
  1491. // Finished with the JVM.
  1492. lJavaVM->DetachCurrentThread();
  1493. return false;
  1494. }
  1495. }
  1496. U32 android_GetFileSize(const char* pFilePath)
  1497. {
  1498. // Attaches the current thread to the JVM.
  1499. /*jint lResult;
  1500. jint lFlags = 0;
  1501. JavaVM* lJavaVM = engine.app->activity->vm;
  1502. JNIEnv* lJNIEnv = engine.app->activity->env;
  1503. JavaVMAttachArgs lJavaVMAttachArgs;
  1504. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1505. lJavaVMAttachArgs.name = "NativeThread";
  1506. lJavaVMAttachArgs.group = NULL;
  1507. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1508. if (lResult == JNI_ERR) {
  1509. return false;
  1510. }
  1511. // Retrieves NativeActivity.
  1512. jobject lNativeActivity = engine.app->activity->clazz;
  1513. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1514. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1515. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1516. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1517. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1518. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1519. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1520. jstring strFileName = lJNIEnv->NewStringUTF(pFilePath);
  1521. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "GetFileSize", "(Landroid/content/Context;Ljava/lang/String;)I");
  1522. jint jsize = lJNIEnv->CallStaticIntMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strFileName);
  1523. long size = jsize;
  1524. lJNIEnv->DeleteLocalRef(strClassName);
  1525. lJNIEnv->DeleteLocalRef(strFileName);
  1526. // Finished with the JVM.
  1527. lJavaVM->DetachCurrentThread();
  1528. return size;
  1529. */
  1530. return 0;
  1531. }
  1532. ConsoleFunction(dumpFontList, void, 1, 1, "Print device fonts to console")
  1533. {
  1534. activity.dumpFontList();
  1535. }
  1536. ConsoleFunction(hideSplashScreen, void, 1, 1, "hide the splash screen")
  1537. {
  1538. toggleSplashScreen(false);
  1539. }
  1540. ConsoleFunction(GetAndroidResolution, const char*, 1, 1, "Returns the resolution for the android device")
  1541. {
  1542. S32 width = _AndroidGetScreenWidth();
  1543. S32 height = _AndroidGetScreenHeight();
  1544. S32 bitdepth = ANDROID_DEFAULT_RESOLUTION_BIT_DEPTH;
  1545. char buf[80];
  1546. dSprintf(buf, sizeof(buf), "%d %d %d", width, height, bitdepth);
  1547. char* buffer = Con::getReturnBuffer(strlen(buf) + 1);
  1548. dStrcpy(buffer, buf);
  1549. return buffer;
  1550. }
  1551. bool Platform::openWebBrowser(const char *webAddress)
  1552. {
  1553. // Attaches the current thread to the JVM.
  1554. jint lResult;
  1555. jint lFlags = 0;
  1556. JavaVM* lJavaVM = engine.app->activity->vm;
  1557. JNIEnv* lJNIEnv = engine.app->activity->env;
  1558. JavaVMAttachArgs lJavaVMAttachArgs;
  1559. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1560. lJavaVMAttachArgs.name = "NativeThread";
  1561. lJavaVMAttachArgs.group = NULL;
  1562. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1563. if (lResult == JNI_ERR) {
  1564. return false;
  1565. }
  1566. // Retrieves NativeActivity.
  1567. jobject lNativeActivity = engine.app->activity->clazz;
  1568. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1569. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1570. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1571. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1572. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1573. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1574. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1575. jstring strURL = lJNIEnv->NewStringUTF(webAddress);
  1576. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "OpenURL", "(Landroid/content/Context;Ljava/lang/String;)V");
  1577. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strURL);
  1578. lJNIEnv->DeleteLocalRef(strClassName);
  1579. lJNIEnv->DeleteLocalRef(strURL);
  1580. // Finished with the JVM.
  1581. lJavaVM->DetachCurrentThread();
  1582. return true;
  1583. }
  1584. void android_AlertOK(const char *title, const char *message)
  1585. {
  1586. // Attaches the current thread to the JVM.
  1587. jint lResult;
  1588. jint lFlags = 0;
  1589. JavaVM* lJavaVM = engine.app->activity->vm;
  1590. JNIEnv* lJNIEnv = engine.app->activity->env;
  1591. JavaVMAttachArgs lJavaVMAttachArgs;
  1592. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1593. lJavaVMAttachArgs.name = "NativeThread";
  1594. lJavaVMAttachArgs.group = NULL;
  1595. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1596. if (lResult == JNI_ERR) {
  1597. return;
  1598. }
  1599. // Retrieves NativeActivity.
  1600. jobject lNativeActivity = engine.app->activity->clazz;
  1601. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1602. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1603. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1604. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1605. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1606. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1607. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1608. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1609. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1610. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertOK", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1611. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1612. lJNIEnv->DeleteLocalRef(strClassName);
  1613. lJNIEnv->DeleteLocalRef(strTitle);
  1614. lJNIEnv->DeleteLocalRef(strMessage);
  1615. // Finished with the JVM.
  1616. lJavaVM->DetachCurrentThread();
  1617. }
  1618. int android_checkAlert()
  1619. {
  1620. // Attaches the current thread to the JVM.
  1621. jint lResult;
  1622. jint lFlags = 0;
  1623. JavaVM* lJavaVM = engine.app->activity->vm;
  1624. JNIEnv* lJNIEnv = engine.app->activity->env;
  1625. JavaVMAttachArgs lJavaVMAttachArgs;
  1626. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1627. lJavaVMAttachArgs.name = "NativeThread";
  1628. lJavaVMAttachArgs.group = NULL;
  1629. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1630. if (lResult == JNI_ERR) {
  1631. return 0;
  1632. }
  1633. // Retrieves NativeActivity.
  1634. jobject lNativeActivity = engine.app->activity->clazz;
  1635. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1636. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1637. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1638. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1639. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1640. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1641. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1642. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "CheckAlert", "()I");
  1643. jint jret = lJNIEnv->CallStaticIntMethod(T2DUtilitiesClass, MethodT2DUtilities);
  1644. int ret = jret;
  1645. lJNIEnv->DeleteLocalRef(strClassName);
  1646. // Finished with the JVM.
  1647. lJavaVM->DetachCurrentThread();
  1648. return ret;
  1649. }
  1650. void android_AlertOKCancel(const char *title, const char *message)
  1651. {
  1652. // Attaches the current thread to the JVM.
  1653. jint lResult;
  1654. jint lFlags = 0;
  1655. JavaVM* lJavaVM = engine.app->activity->vm;
  1656. JNIEnv* lJNIEnv = engine.app->activity->env;
  1657. JavaVMAttachArgs lJavaVMAttachArgs;
  1658. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1659. lJavaVMAttachArgs.name = "NativeThread";
  1660. lJavaVMAttachArgs.group = NULL;
  1661. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1662. if (lResult == JNI_ERR) {
  1663. return;
  1664. }
  1665. // Retrieves NativeActivity.
  1666. jobject lNativeActivity = engine.app->activity->clazz;
  1667. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1668. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1669. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1670. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1671. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1672. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1673. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1674. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1675. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1676. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertOKCancel", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1677. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1678. lJNIEnv->DeleteLocalRef(strClassName);
  1679. lJNIEnv->DeleteLocalRef(strTitle);
  1680. lJNIEnv->DeleteLocalRef(strMessage);
  1681. // Finished with the JVM.
  1682. lJavaVM->DetachCurrentThread();
  1683. }
  1684. void android_AlertRetry(const char *title, const char *message)
  1685. {
  1686. // Attaches the current thread to the JVM.
  1687. jint lResult;
  1688. jint lFlags = 0;
  1689. JavaVM* lJavaVM = engine.app->activity->vm;
  1690. JNIEnv* lJNIEnv = engine.app->activity->env;
  1691. JavaVMAttachArgs lJavaVMAttachArgs;
  1692. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1693. lJavaVMAttachArgs.name = "NativeThread";
  1694. lJavaVMAttachArgs.group = NULL;
  1695. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1696. if (lResult == JNI_ERR) {
  1697. return;
  1698. }
  1699. // Retrieves NativeActivity.
  1700. jobject lNativeActivity = engine.app->activity->clazz;
  1701. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1702. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1703. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1704. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1705. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1706. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1707. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1708. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1709. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1710. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertRetry", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1711. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1712. lJNIEnv->DeleteLocalRef(strClassName);
  1713. lJNIEnv->DeleteLocalRef(strTitle);
  1714. lJNIEnv->DeleteLocalRef(strMessage);
  1715. // Finished with the JVM.
  1716. lJavaVM->DetachCurrentThread();
  1717. }
  1718. void android_AlertYesNo(const char *title, const char *message)
  1719. {
  1720. // Attaches the current thread to the JVM.
  1721. jint lResult;
  1722. jint lFlags = 0;
  1723. JavaVM* lJavaVM = engine.app->activity->vm;
  1724. JNIEnv* lJNIEnv = engine.app->activity->env;
  1725. JavaVMAttachArgs lJavaVMAttachArgs;
  1726. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1727. lJavaVMAttachArgs.name = "NativeThread";
  1728. lJavaVMAttachArgs.group = NULL;
  1729. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1730. if (lResult == JNI_ERR) {
  1731. return;
  1732. }
  1733. // Retrieves NativeActivity.
  1734. jobject lNativeActivity = engine.app->activity->clazz;
  1735. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1736. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1737. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1738. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1739. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1740. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1741. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1742. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1743. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1744. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertYesNo", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1745. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1746. lJNIEnv->DeleteLocalRef(strClassName);
  1747. lJNIEnv->DeleteLocalRef(strTitle);
  1748. lJNIEnv->DeleteLocalRef(strMessage);
  1749. // Finished with the JVM.
  1750. lJavaVM->DetachCurrentThread();
  1751. }
  1752. void android_LoadMusicTrack( const char *mFilename )
  1753. {
  1754. // Attaches the current thread to the JVM.
  1755. jint lResult;
  1756. jint lFlags = 0;
  1757. JavaVM* lJavaVM = engine.app->activity->vm;
  1758. JNIEnv* lJNIEnv = engine.app->activity->env;
  1759. JavaVMAttachArgs lJavaVMAttachArgs;
  1760. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1761. lJavaVMAttachArgs.name = "NativeThread";
  1762. lJavaVMAttachArgs.group = NULL;
  1763. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1764. if (lResult == JNI_ERR) {
  1765. return;
  1766. }
  1767. // Retrieves NativeActivity.
  1768. jobject lNativeActivity = engine.app->activity->clazz;
  1769. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1770. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1771. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1772. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1773. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1774. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1775. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1776. jstring strFilename = lJNIEnv->NewStringUTF(mFilename);
  1777. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "LoadMusicTrack", "(Landroid/content/Context;Ljava/lang/String;)V");
  1778. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer, lNativeActivity, strFilename);
  1779. lJNIEnv->DeleteLocalRef(strClassName);
  1780. lJNIEnv->DeleteLocalRef(strFilename);
  1781. // Finished with the JVM.
  1782. lJavaVM->DetachCurrentThread();
  1783. }
  1784. void android_UnLoadMusicTrack()
  1785. {
  1786. // Attaches the current thread to the JVM.
  1787. jint lResult;
  1788. jint lFlags = 0;
  1789. JavaVM* lJavaVM = engine.app->activity->vm;
  1790. JNIEnv* lJNIEnv = engine.app->activity->env;
  1791. JavaVMAttachArgs lJavaVMAttachArgs;
  1792. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1793. lJavaVMAttachArgs.name = "NativeThread";
  1794. lJavaVMAttachArgs.group = NULL;
  1795. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1796. if (lResult == JNI_ERR) {
  1797. return;
  1798. }
  1799. // Retrieves NativeActivity.
  1800. jobject lNativeActivity = engine.app->activity->clazz;
  1801. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1802. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1803. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1804. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1805. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1806. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1807. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1808. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "UnLoadMusicTrack", "()V");
  1809. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1810. lJNIEnv->DeleteLocalRef(strClassName);
  1811. // Finished with the JVM.
  1812. lJavaVM->DetachCurrentThread();
  1813. }
  1814. bool android_isMusicTrackPlaying()
  1815. {
  1816. // Attaches the current thread to the JVM.
  1817. jint lResult;
  1818. jint lFlags = 0;
  1819. JavaVM* lJavaVM = engine.app->activity->vm;
  1820. JNIEnv* lJNIEnv = engine.app->activity->env;
  1821. JavaVMAttachArgs lJavaVMAttachArgs;
  1822. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1823. lJavaVMAttachArgs.name = "NativeThread";
  1824. lJavaVMAttachArgs.group = NULL;
  1825. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1826. if (lResult == JNI_ERR) {
  1827. return false;
  1828. }
  1829. // Retrieves NativeActivity.
  1830. jobject lNativeActivity = engine.app->activity->clazz;
  1831. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1832. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1833. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1834. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1835. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1836. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1837. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1838. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "isMusicTrackPlaying", "()Z");
  1839. jboolean jret = lJNIEnv->CallStaticBooleanMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1840. lJNIEnv->DeleteLocalRef(strClassName);
  1841. bool ret = jret;
  1842. // Finished with the JVM.
  1843. lJavaVM->DetachCurrentThread();
  1844. return ret;
  1845. }
  1846. void android_StartMusicTrack()
  1847. {
  1848. // Attaches the current thread to the JVM.
  1849. jint lResult;
  1850. jint lFlags = 0;
  1851. JavaVM* lJavaVM = engine.app->activity->vm;
  1852. JNIEnv* lJNIEnv = engine.app->activity->env;
  1853. JavaVMAttachArgs lJavaVMAttachArgs;
  1854. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1855. lJavaVMAttachArgs.name = "NativeThread";
  1856. lJavaVMAttachArgs.group = NULL;
  1857. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1858. if (lResult == JNI_ERR) {
  1859. return;
  1860. }
  1861. // Retrieves NativeActivity.
  1862. jobject lNativeActivity = engine.app->activity->clazz;
  1863. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1864. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1865. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1866. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1867. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1868. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1869. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1870. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "startMusicTrack", "()V");
  1871. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1872. lJNIEnv->DeleteLocalRef(strClassName);
  1873. // Finished with the JVM.
  1874. lJavaVM->DetachCurrentThread();
  1875. }
  1876. void android_StopMusicTrack()
  1877. {
  1878. // Attaches the current thread to the JVM.
  1879. jint lResult;
  1880. jint lFlags = 0;
  1881. JavaVM* lJavaVM = engine.app->activity->vm;
  1882. JNIEnv* lJNIEnv = engine.app->activity->env;
  1883. JavaVMAttachArgs lJavaVMAttachArgs;
  1884. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1885. lJavaVMAttachArgs.name = "NativeThread";
  1886. lJavaVMAttachArgs.group = NULL;
  1887. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1888. if (lResult == JNI_ERR) {
  1889. return;
  1890. }
  1891. // Retrieves NativeActivity.
  1892. jobject lNativeActivity = engine.app->activity->clazz;
  1893. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1894. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1895. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1896. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1897. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1898. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1899. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1900. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "stopMusicTrack", "()V");
  1901. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1902. lJNIEnv->DeleteLocalRef(strClassName);
  1903. // Finished with the JVM.
  1904. lJavaVM->DetachCurrentThread();
  1905. }
  1906. void android_setMusicTrackVolume(F32 volume)
  1907. {
  1908. // Attaches the current thread to the JVM.
  1909. jint lResult;
  1910. jint lFlags = 0;
  1911. JavaVM* lJavaVM = engine.app->activity->vm;
  1912. JNIEnv* lJNIEnv = engine.app->activity->env;
  1913. JavaVMAttachArgs lJavaVMAttachArgs;
  1914. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1915. lJavaVMAttachArgs.name = "NativeThread";
  1916. lJavaVMAttachArgs.group = NULL;
  1917. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1918. if (lResult == JNI_ERR) {
  1919. return;
  1920. }
  1921. // Retrieves NativeActivity.
  1922. jobject lNativeActivity = engine.app->activity->clazz;
  1923. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1924. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1925. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1926. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1927. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1928. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1929. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1930. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "setMusicTrackVolume", "(F)V");
  1931. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer, (jfloat)volume);
  1932. lJNIEnv->DeleteLocalRef(strClassName);
  1933. // Finished with the JVM.
  1934. lJavaVM->DetachCurrentThread();
  1935. }
  1936. ConsoleFunction(doDeviceVibrate, void, 1, 1, "Makes the device do a quick vibration. Only works on devices with vibration functionality.")
  1937. {
  1938. // Vibrate for 500 milliseconds
  1939. long vibrateTimeMS = 500L;
  1940. // Attaches the current thread to the JVM.
  1941. jint lResult;
  1942. jint lFlags = 0;
  1943. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  1944. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  1945. JavaVMAttachArgs lJavaVMAttachArgs;
  1946. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1947. lJavaVMAttachArgs.name = "NativeThread";
  1948. lJavaVMAttachArgs.group = NULL;
  1949. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1950. if (lResult == JNI_ERR) {
  1951. return;
  1952. }
  1953. // Retrieves NativeActivity.
  1954. jobject lNativeActivity = platState.engine->app->activity->clazz;
  1955. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1956. // Retrieves Context.VIBRATOR_SERVICE.
  1957. jclass ClassContext = lJNIEnv->FindClass("android/content/Context");
  1958. jfieldID FieldVIBRATOR_SERVICE =lJNIEnv->GetStaticFieldID(ClassContext,"VIBRATOR_SERVICE", "Ljava/lang/String;");
  1959. jobject VIBRATOR_SERVICE = lJNIEnv->GetStaticObjectField(ClassContext, FieldVIBRATOR_SERVICE);
  1960. // Runs getSystemService(VIBRATOR_SERVICE)
  1961. jmethodID MethodGetSystemService = lJNIEnv->GetMethodID(ClassNativeActivity, "getSystemService","(Ljava/lang/String;)Ljava/lang/Object;");
  1962. jobject lVibrator = lJNIEnv->CallObjectMethod(lNativeActivity, MethodGetSystemService, VIBRATOR_SERVICE);
  1963. //Runs v.vibrate(ms)
  1964. jclass ClassVibrator = lJNIEnv->FindClass("android/os/Vibrator");
  1965. jmethodID MethodVibrate = lJNIEnv->GetMethodID(ClassVibrator, "vibrate", "(J)V");
  1966. lJNIEnv->CallVoidMethod(lVibrator, MethodVibrate, vibrateTimeMS);
  1967. // Finished with the JVM.
  1968. lJavaVM->DetachCurrentThread();
  1969. }
  1970. ConsoleFunction(enableAccelerometer, void, 1, 1, "() Allow accelerometer tracking during device motion updates")
  1971. {
  1972. activity.enableAccelerometer(true);
  1973. }
  1974. ConsoleFunction(disableAccelerometer, void, 1, 1, "() Stop accelerometer tracking")
  1975. {
  1976. activity.enableAccelerometer(false);
  1977. }
  1978. ConsoleFunction(isAccelerometerActive, bool, 1, 1, "() Check to see if Accelerometer is being polled\n"
  1979. "@return True if accelerometer is on, false otherwise")
  1980. {
  1981. return activity.isAccelerometerActive();
  1982. }
  1983. void adprintf(const char* fmt,...) {
  1984. va_list argptr;
  1985. int cnt;
  1986. char s[4096];
  1987. time_t now;
  1988. va_start(argptr,fmt);
  1989. cnt = vsprintf(s,fmt,argptr);
  1990. va_end(argptr);
  1991. time(&now);
  1992. tm* t = localtime(&now);
  1993. std::stringstream ss;
  1994. ss.clear();
  1995. ss << "[";
  1996. ss << (t->tm_year + 1900);
  1997. ss << "/";
  1998. if (t->tm_mon < 9)
  1999. ss << "0";
  2000. ss << (t->tm_mon+1);
  2001. ss << "/";
  2002. if (t->tm_mday < 10)
  2003. ss << "0";
  2004. ss << t->tm_mday;
  2005. ss << " ";
  2006. if (t->tm_hour < 10)
  2007. ss << "0";
  2008. ss << t->tm_hour;
  2009. ss << ":";
  2010. if (t->tm_min < 10)
  2011. ss << "0";
  2012. ss << t->tm_min;
  2013. ss << ":";
  2014. if (t->tm_sec < 10)
  2015. ss << "0";
  2016. ss << t->tm_sec;
  2017. ss << "] ";
  2018. ss << s;
  2019. __android_log_print(ANDROID_LOG_INFO, "Torque2D", "%s", ss.str().c_str());
  2020. }