T2DActivity.cpp 81 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493
  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. toggleSplashScreen(true);
  574. if (SetupCompleted == false)
  575. {
  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. toggleSplashScreen(false);
  857. }
  858. break;
  859. case APP_CMD_TERM_WINDOW:
  860. bSuspended = true;
  861. Game->textureKill();
  862. // The window is being hidden or closed, clean it up.
  863. engine_term_display(engine, false);
  864. break;
  865. case APP_CMD_RESUME:
  866. break;
  867. case APP_CMD_GAINED_FOCUS:
  868. // When our app gains focus, we start monitoring the accelerometer.
  869. if (engine->accelerometerSensor != NULL && activity.isAccelerometerActive()) {
  870. ASensorEventQueue_enableSensor(engine->sensorEventQueue,
  871. engine->accelerometerSensor);
  872. // We'd like to get 30 events per second (in us).
  873. ASensorEventQueue_setEventRate(engine->sensorEventQueue,
  874. engine->accelerometerSensor, (1000L/30)*1000);
  875. }
  876. engine->animating = 1;
  877. break;
  878. case APP_CMD_LOST_FOCUS:
  879. // When our app loses focus, we stop monitoring the accelerometer.
  880. // This is to avoid consuming battery while not being used.
  881. if (engine->accelerometerSensor != NULL) {
  882. ASensorEventQueue_disableSensor(engine->sensorEventQueue,
  883. engine->accelerometerSensor);
  884. }
  885. // stop animating.
  886. engine->animating = 0;
  887. break;
  888. case APP_CMD_PAUSE:
  889. break;
  890. }
  891. }
  892. struct engine engine;
  893. /**
  894. * This is the main entry point of a native application that is using
  895. * android_native_app_glue. It runs in its own thread, with its own
  896. * event loop for receiving input events and doing other things.
  897. */
  898. void android_main(struct android_app* state) {
  899. //init startup time so U32 doesnt overflow
  900. android_StartupTime();
  901. // Make sure glue isn't stripped.
  902. app_dummy();
  903. platState.engine = &engine;
  904. memset(&engine, 0, sizeof(engine));
  905. state->userData = &engine;
  906. state->onAppCmd = engine_handle_cmd;
  907. state->onInputEvent = engine_handle_input;
  908. engine.app = state;
  909. // Prepare to monitor accelerometer
  910. engine.sensorManager = ASensorManager_getInstance();
  911. engine.accelerometerSensor = ASensorManager_getDefaultSensor(engine.sensorManager,
  912. ASENSOR_TYPE_ACCELEROMETER);
  913. engine.sensorEventQueue = ASensorManager_createEventQueue(engine.sensorManager,
  914. state->looper, LOOPER_ID_USER, NULL, NULL);
  915. if (state->savedState != NULL) {
  916. // We are starting with a previous saved state; restore from it.
  917. engine.state = *(struct saved_state*)state->savedState;
  918. }
  919. keepScreenOn();
  920. //This is to help the debugger catch up. If you dont have this you cant debug this early in the execution
  921. //so only uncomment when needed as it adds 10 seconds to startup
  922. //sleep(10);
  923. //store the cache dir
  924. activity.loadCacheDir();
  925. //enumerate fonts
  926. activity.enumerateFonts();
  927. platState.argc = 0;
  928. // loop waiting for stuff to do.
  929. while (1) {
  930. // Read all pending events.
  931. int ident;
  932. int events;
  933. struct android_poll_source* source;
  934. // If not animating, we will block forever waiting for events.
  935. // If animating, we loop until all events are read, then continue
  936. // to draw the next frame of animation.
  937. while ((ident=ALooper_pollAll(engine.animating ? 0 : -1, NULL, &events,
  938. (void**)&source)) >= 0) {
  939. // Process this event.
  940. if (source != NULL) {
  941. source->process(state, source);
  942. }
  943. // If a sensor has data, process it now.
  944. if (ident == LOOPER_ID_USER) {
  945. if (engine.accelerometerSensor != NULL && activity.isAccelerometerActive()) {
  946. ASensorEvent event;
  947. int i = 0;
  948. while (ASensorEventQueue_getEvents(engine.sensorEventQueue,
  949. &event, 1) > 0) {
  950. InputEvent inputEvent;
  951. U32 accelAxes[6] = { SI_ACCELX, SI_ACCELY, SI_ACCELZ, SI_GRAVX, SI_GRAVY, SI_GRAVZ };
  952. double userAcc[6] = { event.acceleration.x, event.acceleration.y, event.acceleration.z,0,0,0};
  953. inputEvent.deviceInst = 0;
  954. inputEvent.fValue = userAcc[i];
  955. inputEvent.deviceType = AccelerometerDeviceType;
  956. inputEvent.objType = accelAxes[i];
  957. inputEvent.objInst = i;
  958. inputEvent.action = SI_MOTION;
  959. inputEvent.modifier = 0;
  960. Game->postEvent(inputEvent);
  961. i++;
  962. }
  963. }
  964. }
  965. // Check if we are exiting.
  966. if (state->destroyRequested != 0) {
  967. engine_term_display(&engine, true);
  968. return;
  969. }
  970. }
  971. engine_update_frame(&engine);
  972. /*if (engine.animating) {
  973. // Drawing is throttled to the screen update rate, so there
  974. // is no need to do timing here.
  975. engine_draw_frame(&engine);
  976. }*/
  977. }
  978. engine_term_display(&engine, true);
  979. }
  980. struct MatchPathSeparator
  981. {
  982. bool operator()( char ch ) const
  983. {
  984. return ch == '/';
  985. }
  986. };
  987. bool android_DumpDirectoriesExtra(Vector<StringTableEntry> &directoryVector)
  988. {
  989. // Attaches the current thread to the JVM.
  990. jint lResult;
  991. jint lFlags = 0;
  992. JavaVM* lJavaVM = engine.app->activity->vm;
  993. JNIEnv* lJNIEnv = engine.app->activity->env;
  994. JavaVMAttachArgs lJavaVMAttachArgs;
  995. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  996. lJavaVMAttachArgs.name = "NativeThread";
  997. lJavaVMAttachArgs.group = NULL;
  998. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  999. if (lResult == JNI_ERR) {
  1000. return false;
  1001. }
  1002. // Retrieves NativeActivity.
  1003. jobject lNativeActivity = engine.app->activity->clazz;
  1004. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1005. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1006. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1007. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1008. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1009. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1010. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1011. jmethodID MethodExtraPaths = lJNIEnv->GetStaticMethodID(FileWalkerClass, "getRestOfDump", "()[Ljava/lang/String;");
  1012. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodExtraPaths);
  1013. bool ret = true;
  1014. if (stringArray)
  1015. {
  1016. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1017. if (stringCount < 500)
  1018. ret = false;
  1019. for (int i=0; i<stringCount; i++) {
  1020. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1021. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1022. char str[255];
  1023. strcpy(str, rawString);
  1024. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1025. if (!Platform::isExcludedDirectory(str))
  1026. {
  1027. directoryVector.push_back(StringTable->insert(str));
  1028. }
  1029. }
  1030. lJNIEnv->DeleteLocalRef(stringArray);
  1031. }
  1032. else
  1033. {
  1034. ret = false;
  1035. }
  1036. lJNIEnv->DeleteLocalRef(strClassName);
  1037. // Finished with the JVM.
  1038. lJavaVM->DetachCurrentThread();
  1039. return ret;
  1040. }
  1041. static Vector<Platform::FileInfo> dumpPathBackup;
  1042. bool android_DumpDirectories(const char *basePath, const char *path, Vector<StringTableEntry> &directoryVector, S32 depth, bool noBasePath)
  1043. {
  1044. // Attaches the current thread to the JVM.
  1045. jint lResult;
  1046. jint lFlags = 0;
  1047. JavaVM* lJavaVM = engine.app->activity->vm;
  1048. JNIEnv* lJNIEnv = engine.app->activity->env;
  1049. JavaVMAttachArgs lJavaVMAttachArgs;
  1050. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1051. lJavaVMAttachArgs.name = "NativeThread";
  1052. lJavaVMAttachArgs.group = NULL;
  1053. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1054. if (lResult == JNI_ERR) {
  1055. return false;
  1056. }
  1057. // Retrieves NativeActivity.
  1058. jobject lNativeActivity = engine.app->activity->clazz;
  1059. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1060. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1061. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1062. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1063. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1064. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1065. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1066. jstring strDirName = lJNIEnv->NewStringUTF(basePath);
  1067. jstring strDirName2 = lJNIEnv->NewStringUTF(path);
  1068. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "DumpDirectories", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;ZZ)[Ljava/lang/String;");
  1069. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName, strDirName2, depth != 0, noBasePath);
  1070. bool ret = false;
  1071. if (stringArray)
  1072. {
  1073. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1074. for (int i=0; i<stringCount; i++) {
  1075. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1076. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1077. char str[255];
  1078. strcpy(str, rawString);
  1079. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1080. if (!Platform::isExcludedDirectory(str))
  1081. {
  1082. directoryVector.push_back(StringTable->insert(str));
  1083. }
  1084. }
  1085. lJNIEnv->DeleteLocalRef(stringArray);
  1086. bool keepGoing = false;
  1087. if (stringCount == 500)
  1088. {
  1089. keepGoing = true;
  1090. }
  1091. lJNIEnv->DeleteLocalRef(strClassName);
  1092. lJNIEnv->DeleteLocalRef(strDirName);
  1093. lJNIEnv->DeleteLocalRef(strDirName2);
  1094. // Finished with the JVM.
  1095. lJavaVM->DetachCurrentThread();
  1096. while (keepGoing == true)
  1097. keepGoing = android_DumpDirectoriesExtra(directoryVector);
  1098. ret = true;
  1099. }
  1100. else
  1101. {
  1102. lJNIEnv->DeleteLocalRef(strClassName);
  1103. lJNIEnv->DeleteLocalRef(strDirName);
  1104. lJNIEnv->DeleteLocalRef(strDirName2);
  1105. // Finished with the JVM.
  1106. lJavaVM->DetachCurrentThread();
  1107. ret = false;
  1108. }
  1109. return ret;
  1110. }
  1111. bool android_DumpPathExtra(Vector<Platform::FileInfo>& fileVector)
  1112. {
  1113. // Attaches the current thread to the JVM.
  1114. jint lResult;
  1115. jint lFlags = 0;
  1116. JavaVM* lJavaVM = engine.app->activity->vm;
  1117. JNIEnv* lJNIEnv = engine.app->activity->env;
  1118. JavaVMAttachArgs lJavaVMAttachArgs;
  1119. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1120. lJavaVMAttachArgs.name = "NativeThread";
  1121. lJavaVMAttachArgs.group = NULL;
  1122. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1123. if (lResult == JNI_ERR) {
  1124. return false;
  1125. }
  1126. // Retrieves NativeActivity.
  1127. jobject lNativeActivity = engine.app->activity->clazz;
  1128. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1129. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1130. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1131. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1132. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1133. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1134. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1135. jmethodID MethodExtraPaths = lJNIEnv->GetStaticMethodID(FileWalkerClass, "getRestOfDump", "()[Ljava/lang/String;");
  1136. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodExtraPaths);
  1137. bool ret = true;
  1138. if (stringArray)
  1139. {
  1140. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1141. if (stringCount < 500)
  1142. ret = false;
  1143. for (int i=0; i<stringCount; i++) {
  1144. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1145. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1146. std::string str = rawString;
  1147. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1148. const U32 fileSize = Platform::getFileSize(str.c_str());
  1149. fileVector.increment();
  1150. Platform::FileInfo& rInfo = fileVector.last();
  1151. std::string fileName = std::string(
  1152. std::find_if( str.rbegin(), str.rend(),
  1153. MatchPathSeparator() ).base(),
  1154. str.end() );
  1155. rInfo.pFullPath = StringTable->insert(str.substr(0,str.find(fileName)-1).c_str());
  1156. rInfo.pFileName = StringTable->insert(fileName.c_str());
  1157. rInfo.fileSize = fileSize;
  1158. dumpPathBackup.increment();
  1159. Platform::FileInfo& rInfo2 = dumpPathBackup.last();
  1160. rInfo2.pFullPath = rInfo.pFullPath;
  1161. rInfo2.pFileName = rInfo.pFileName;
  1162. rInfo2.fileSize = rInfo.fileSize;
  1163. }
  1164. lJNIEnv->DeleteLocalRef(stringArray);
  1165. }
  1166. else
  1167. {
  1168. ret = false;
  1169. }
  1170. lJNIEnv->DeleteLocalRef(strClassName);
  1171. // Finished with the JVM.
  1172. lJavaVM->DetachCurrentThread();
  1173. return ret;
  1174. }
  1175. bool loadDumpPathFromCache(const char* dir, Vector<Platform::FileInfo>& fileVector, U32 depth)
  1176. {
  1177. bool foundPaths = false;
  1178. for (int i = 0; i < dumpPathBackup.size(); i++)
  1179. {
  1180. if (depth == 0)
  1181. {
  1182. Platform::FileInfo fi = dumpPathBackup[i];
  1183. if (strcmp(fi.pFullPath, dir) == 0)
  1184. {
  1185. fileVector.increment();
  1186. Platform::FileInfo& rInfo = fileVector.last();
  1187. rInfo.pFileName = fi.pFileName;
  1188. rInfo.pFullPath = fi.pFullPath;
  1189. rInfo.fileSize = fi.fileSize;
  1190. foundPaths = true;
  1191. }
  1192. }
  1193. else
  1194. {
  1195. Platform::FileInfo fi = dumpPathBackup[i];
  1196. std::string fullPath = fi.pFullPath;
  1197. if (fullPath.find(dir) == 0)
  1198. {
  1199. fileVector.increment();
  1200. Platform::FileInfo& rInfo = fileVector.last();
  1201. rInfo.pFileName = fi.pFileName;
  1202. rInfo.pFullPath = fi.pFullPath;
  1203. rInfo.fileSize = fi.fileSize;
  1204. foundPaths = true;
  1205. }
  1206. }
  1207. }
  1208. return true;
  1209. }
  1210. bool android_DumpPath(const char* dir, Vector<Platform::FileInfo>& fileVector, U32 depth)
  1211. {
  1212. if (dumpPathBackup.size() > 0)
  1213. {
  1214. return loadDumpPathFromCache(dir, fileVector, depth);
  1215. }
  1216. // Attaches the current thread to the JVM.
  1217. jint lResult;
  1218. jint lFlags = 0;
  1219. JavaVM* lJavaVM = engine.app->activity->vm;
  1220. JNIEnv* lJNIEnv = engine.app->activity->env;
  1221. JavaVMAttachArgs lJavaVMAttachArgs;
  1222. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1223. lJavaVMAttachArgs.name = "NativeThread";
  1224. lJavaVMAttachArgs.group = NULL;
  1225. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1226. if (lResult == JNI_ERR) {
  1227. return false;
  1228. }
  1229. // Retrieves NativeActivity.
  1230. jobject lNativeActivity = engine.app->activity->clazz;
  1231. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1232. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1233. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1234. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1235. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1236. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1237. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1238. jstring strDirName = lJNIEnv->NewStringUTF(dir);
  1239. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "DumpPath", "(Landroid/content/Context;Ljava/lang/String;Z)[Ljava/lang/String;");
  1240. jobjectArray stringArray = (jobjectArray)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName, depth != 0);
  1241. bool ret = false;
  1242. if (stringArray)
  1243. {
  1244. int stringCount = lJNIEnv->GetArrayLength(stringArray);
  1245. for (int i=0; i<stringCount; i++) {
  1246. jstring string = (jstring) lJNIEnv->GetObjectArrayElement(stringArray, i);
  1247. const char *rawString = lJNIEnv->GetStringUTFChars(string, 0);
  1248. std::string str = rawString;
  1249. lJNIEnv->ReleaseStringUTFChars(string, rawString);
  1250. const U32 fileSize = Platform::getFileSize(str.c_str());
  1251. fileVector.increment();
  1252. Platform::FileInfo& rInfo = fileVector.last();
  1253. std::string fileName = std::string(
  1254. std::find_if( str.rbegin(), str.rend(),
  1255. MatchPathSeparator() ).base(),
  1256. str.end() );
  1257. rInfo.pFullPath = StringTable->insert(str.substr(0,str.find(fileName)-1).c_str());
  1258. rInfo.pFileName = StringTable->insert(fileName.c_str());
  1259. rInfo.fileSize = fileSize;
  1260. dumpPathBackup.increment();
  1261. Platform::FileInfo& rInfo2 = dumpPathBackup.last();
  1262. rInfo2.pFullPath = rInfo.pFullPath;
  1263. rInfo2.pFileName = rInfo.pFileName;
  1264. rInfo2.fileSize = rInfo.fileSize;
  1265. }
  1266. lJNIEnv->DeleteLocalRef(stringArray);
  1267. stringArray = NULL;
  1268. lJNIEnv->DeleteLocalRef(strClassName);
  1269. lJNIEnv->DeleteLocalRef(strDirName);
  1270. // Finished with the JVM.
  1271. lJavaVM->DetachCurrentThread();
  1272. bool keepGoing = false;
  1273. if (stringCount == 500)
  1274. {
  1275. keepGoing = true;
  1276. }
  1277. while (keepGoing == true)
  1278. keepGoing = android_DumpPathExtra(fileVector);
  1279. ret = true;
  1280. }
  1281. else
  1282. {
  1283. lJNIEnv->DeleteLocalRef(strClassName);
  1284. lJNIEnv->DeleteLocalRef(strDirName);
  1285. // Finished with the JVM.
  1286. lJavaVM->DetachCurrentThread();
  1287. ret = false;
  1288. }
  1289. return ret;
  1290. }
  1291. void android_InitDirList(const char* dir)
  1292. {
  1293. // Attaches the current thread to the JVM.
  1294. jint lResult;
  1295. jint lFlags = 0;
  1296. JavaVM* lJavaVM = engine.app->activity->vm;
  1297. JNIEnv* lJNIEnv = engine.app->activity->env;
  1298. JavaVMAttachArgs lJavaVMAttachArgs;
  1299. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1300. lJavaVMAttachArgs.name = "NativeThread";
  1301. lJavaVMAttachArgs.group = NULL;
  1302. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1303. if (lResult == JNI_ERR) {
  1304. return;
  1305. }
  1306. // Retrieves NativeActivity.
  1307. jobject lNativeActivity = engine.app->activity->clazz;
  1308. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1309. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1310. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1311. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1312. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1313. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1314. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1315. jstring strDirName = lJNIEnv->NewStringUTF(dir);
  1316. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "InitDirList", "(Landroid/content/Context;Ljava/lang/String;)V");
  1317. lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName);
  1318. lJNIEnv->DeleteLocalRef(strClassName);
  1319. lJNIEnv->DeleteLocalRef(strDirName);
  1320. // Finished with the JVM.
  1321. lJavaVM->DetachCurrentThread();
  1322. }
  1323. void android_GetNextDir(const char* pdir, char *dir)
  1324. {
  1325. // Attaches the current thread to the JVM.
  1326. jint lResult;
  1327. jint lFlags = 0;
  1328. JavaVM* lJavaVM = engine.app->activity->vm;
  1329. JNIEnv* lJNIEnv = engine.app->activity->env;
  1330. JavaVMAttachArgs lJavaVMAttachArgs;
  1331. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1332. lJavaVMAttachArgs.name = "NativeThread";
  1333. lJavaVMAttachArgs.group = NULL;
  1334. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1335. if (lResult == JNI_ERR) {
  1336. return;
  1337. }
  1338. // Retrieves NativeActivity.
  1339. jobject lNativeActivity = engine.app->activity->clazz;
  1340. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1341. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1342. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1343. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1344. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1345. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1346. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1347. jstring strDirName = lJNIEnv->NewStringUTF(pdir);
  1348. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "GetNextDir", "(Ljava/lang/String;)Ljava/lang/String;");
  1349. jstring jpath = (jstring)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, strDirName);
  1350. if (jpath != NULL)
  1351. {
  1352. const char* path = lJNIEnv->GetStringUTFChars(jpath, NULL);
  1353. strcpy(dir, path);
  1354. dir[strlen(path)] = '\0';
  1355. lJNIEnv->ReleaseStringUTFChars(jpath, path);
  1356. }
  1357. else
  1358. {
  1359. strcpy(dir,"");
  1360. }
  1361. lJNIEnv->DeleteLocalRef(strClassName);
  1362. lJNIEnv->DeleteLocalRef(strDirName);
  1363. // Finished with the JVM.
  1364. lJavaVM->DetachCurrentThread();
  1365. }
  1366. void android_GetNextFile(const char* pdir, char *file)
  1367. {
  1368. // Attaches the current thread to the JVM.
  1369. jint lResult;
  1370. jint lFlags = 0;
  1371. JavaVM* lJavaVM = engine.app->activity->vm;
  1372. JNIEnv* lJNIEnv = engine.app->activity->env;
  1373. JavaVMAttachArgs lJavaVMAttachArgs;
  1374. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1375. lJavaVMAttachArgs.name = "NativeThread";
  1376. lJavaVMAttachArgs.group = NULL;
  1377. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1378. if (lResult == JNI_ERR) {
  1379. return;
  1380. }
  1381. // Retrieves NativeActivity.
  1382. jobject lNativeActivity = engine.app->activity->clazz;
  1383. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1384. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1385. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1386. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1387. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1388. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1389. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1390. jstring strDirName = lJNIEnv->NewStringUTF(pdir);
  1391. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "GetNextFile", "(Ljava/lang/String;)Ljava/lang/String;");
  1392. jstring jpath = (jstring)lJNIEnv->CallStaticObjectMethod(FileWalkerClass, MethodFileWalker, strDirName);
  1393. if (jpath != NULL)
  1394. {
  1395. const char* path = lJNIEnv->GetStringUTFChars(jpath, NULL);
  1396. strcpy(file, path);
  1397. file[strlen(path)] = '\0';
  1398. lJNIEnv->ReleaseStringUTFChars(jpath, path);
  1399. }
  1400. else
  1401. {
  1402. strcpy(file,"");
  1403. }
  1404. lJNIEnv->DeleteLocalRef(strClassName);
  1405. lJNIEnv->DeleteLocalRef(strDirName);
  1406. // Finished with the JVM.
  1407. lJavaVM->DetachCurrentThread();
  1408. }
  1409. bool android_IsFile(const char* path)
  1410. {
  1411. // Attaches the current thread to the JVM.
  1412. jint lResult;
  1413. jint lFlags = 0;
  1414. JavaVM* lJavaVM = engine.app->activity->vm;
  1415. JNIEnv* lJNIEnv = engine.app->activity->env;
  1416. JavaVMAttachArgs lJavaVMAttachArgs;
  1417. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1418. lJavaVMAttachArgs.name = "NativeThread";
  1419. lJavaVMAttachArgs.group = NULL;
  1420. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1421. if (lResult == JNI_ERR) {
  1422. return false;
  1423. }
  1424. // Retrieves NativeActivity.
  1425. jobject lNativeActivity = engine.app->activity->clazz;
  1426. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1427. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1428. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1429. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1430. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1431. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1432. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1433. jstring strDirName = lJNIEnv->NewStringUTF(path);
  1434. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "IsFile", "(Landroid/content/Context;Ljava/lang/String;)Z");
  1435. jboolean jpath = lJNIEnv->CallStaticBooleanMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName);
  1436. if (jpath)
  1437. {
  1438. lJNIEnv->DeleteLocalRef(strClassName);
  1439. lJNIEnv->DeleteLocalRef(strDirName);
  1440. // Finished with the JVM.
  1441. lJavaVM->DetachCurrentThread();
  1442. return true;
  1443. }
  1444. else
  1445. {
  1446. lJNIEnv->DeleteLocalRef(strClassName);
  1447. lJNIEnv->DeleteLocalRef(strDirName);
  1448. // Finished with the JVM.
  1449. lJavaVM->DetachCurrentThread();
  1450. return false;
  1451. }
  1452. }
  1453. bool android_IsDir(const char* path)
  1454. {
  1455. // Attaches the current thread to the JVM.
  1456. jint lResult;
  1457. jint lFlags = 0;
  1458. JavaVM* lJavaVM = engine.app->activity->vm;
  1459. JNIEnv* lJNIEnv = engine.app->activity->env;
  1460. JavaVMAttachArgs lJavaVMAttachArgs;
  1461. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1462. lJavaVMAttachArgs.name = "NativeThread";
  1463. lJavaVMAttachArgs.group = NULL;
  1464. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1465. if (lResult == JNI_ERR) {
  1466. return false;
  1467. }
  1468. // Retrieves NativeActivity.
  1469. jobject lNativeActivity = engine.app->activity->clazz;
  1470. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1471. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1472. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1473. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1474. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1475. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1476. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1477. jstring strDirName = lJNIEnv->NewStringUTF(path);
  1478. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "IsDir", "(Landroid/content/Context;Ljava/lang/String;)Z");
  1479. jboolean jpath = lJNIEnv->CallStaticBooleanMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strDirName);
  1480. if (jpath)
  1481. {
  1482. lJNIEnv->DeleteLocalRef(strClassName);
  1483. lJNIEnv->DeleteLocalRef(strDirName);
  1484. // Finished with the JVM.
  1485. lJavaVM->DetachCurrentThread();
  1486. return true;
  1487. }
  1488. else
  1489. {
  1490. lJNIEnv->DeleteLocalRef(strClassName);
  1491. lJNIEnv->DeleteLocalRef(strDirName);
  1492. // Finished with the JVM.
  1493. lJavaVM->DetachCurrentThread();
  1494. return false;
  1495. }
  1496. }
  1497. U32 android_GetFileSize(const char* pFilePath)
  1498. {
  1499. // Attaches the current thread to the JVM.
  1500. /*jint lResult;
  1501. jint lFlags = 0;
  1502. JavaVM* lJavaVM = engine.app->activity->vm;
  1503. JNIEnv* lJNIEnv = engine.app->activity->env;
  1504. JavaVMAttachArgs lJavaVMAttachArgs;
  1505. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1506. lJavaVMAttachArgs.name = "NativeThread";
  1507. lJavaVMAttachArgs.group = NULL;
  1508. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1509. if (lResult == JNI_ERR) {
  1510. return false;
  1511. }
  1512. // Retrieves NativeActivity.
  1513. jobject lNativeActivity = engine.app->activity->clazz;
  1514. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1515. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1516. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1517. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1518. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1519. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/FileWalker");
  1520. jclass FileWalkerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1521. jstring strFileName = lJNIEnv->NewStringUTF(pFilePath);
  1522. jmethodID MethodFileWalker = lJNIEnv->GetStaticMethodID(FileWalkerClass, "GetFileSize", "(Landroid/content/Context;Ljava/lang/String;)I");
  1523. jint jsize = lJNIEnv->CallStaticIntMethod(FileWalkerClass, MethodFileWalker, lNativeActivity, strFileName);
  1524. long size = jsize;
  1525. lJNIEnv->DeleteLocalRef(strClassName);
  1526. lJNIEnv->DeleteLocalRef(strFileName);
  1527. // Finished with the JVM.
  1528. lJavaVM->DetachCurrentThread();
  1529. return size;
  1530. */
  1531. return 0;
  1532. }
  1533. ConsoleFunction(dumpFontList, void, 1, 1, "Print device fonts to console")
  1534. {
  1535. activity.dumpFontList();
  1536. }
  1537. ConsoleFunction(hideSplashScreen, void, 1, 1, "hide the splash screen")
  1538. {
  1539. toggleSplashScreen(false);
  1540. }
  1541. ConsoleFunction(GetAndroidResolution, const char*, 1, 1, "Returns the resolution for the android device")
  1542. {
  1543. S32 width = _AndroidGetScreenWidth();
  1544. S32 height = _AndroidGetScreenHeight();
  1545. S32 bitdepth = ANDROID_DEFAULT_RESOLUTION_BIT_DEPTH;
  1546. char buf[80];
  1547. dSprintf(buf, sizeof(buf), "%d %d %d", width, height, bitdepth);
  1548. char* buffer = Con::getReturnBuffer(strlen(buf) + 1);
  1549. dStrcpy(buffer, buf);
  1550. return buffer;
  1551. }
  1552. bool Platform::openWebBrowser(const char *webAddress)
  1553. {
  1554. // Attaches the current thread to the JVM.
  1555. jint lResult;
  1556. jint lFlags = 0;
  1557. JavaVM* lJavaVM = engine.app->activity->vm;
  1558. JNIEnv* lJNIEnv = engine.app->activity->env;
  1559. JavaVMAttachArgs lJavaVMAttachArgs;
  1560. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1561. lJavaVMAttachArgs.name = "NativeThread";
  1562. lJavaVMAttachArgs.group = NULL;
  1563. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1564. if (lResult == JNI_ERR) {
  1565. return false;
  1566. }
  1567. // Retrieves NativeActivity.
  1568. jobject lNativeActivity = engine.app->activity->clazz;
  1569. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1570. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1571. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1572. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1573. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1574. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1575. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1576. jstring strURL = lJNIEnv->NewStringUTF(webAddress);
  1577. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "OpenURL", "(Landroid/content/Context;Ljava/lang/String;)V");
  1578. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strURL);
  1579. lJNIEnv->DeleteLocalRef(strClassName);
  1580. lJNIEnv->DeleteLocalRef(strURL);
  1581. // Finished with the JVM.
  1582. lJavaVM->DetachCurrentThread();
  1583. return true;
  1584. }
  1585. void android_AlertOK(const char *title, const char *message)
  1586. {
  1587. // Attaches the current thread to the JVM.
  1588. jint lResult;
  1589. jint lFlags = 0;
  1590. JavaVM* lJavaVM = engine.app->activity->vm;
  1591. JNIEnv* lJNIEnv = engine.app->activity->env;
  1592. JavaVMAttachArgs lJavaVMAttachArgs;
  1593. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1594. lJavaVMAttachArgs.name = "NativeThread";
  1595. lJavaVMAttachArgs.group = NULL;
  1596. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1597. if (lResult == JNI_ERR) {
  1598. return;
  1599. }
  1600. // Retrieves NativeActivity.
  1601. jobject lNativeActivity = engine.app->activity->clazz;
  1602. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1603. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1604. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1605. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1606. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1607. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1608. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1609. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1610. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1611. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertOK", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1612. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1613. lJNIEnv->DeleteLocalRef(strClassName);
  1614. lJNIEnv->DeleteLocalRef(strTitle);
  1615. lJNIEnv->DeleteLocalRef(strMessage);
  1616. // Finished with the JVM.
  1617. lJavaVM->DetachCurrentThread();
  1618. }
  1619. int android_checkAlert()
  1620. {
  1621. // Attaches the current thread to the JVM.
  1622. jint lResult;
  1623. jint lFlags = 0;
  1624. JavaVM* lJavaVM = engine.app->activity->vm;
  1625. JNIEnv* lJNIEnv = engine.app->activity->env;
  1626. JavaVMAttachArgs lJavaVMAttachArgs;
  1627. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1628. lJavaVMAttachArgs.name = "NativeThread";
  1629. lJavaVMAttachArgs.group = NULL;
  1630. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1631. if (lResult == JNI_ERR) {
  1632. return 0;
  1633. }
  1634. // Retrieves NativeActivity.
  1635. jobject lNativeActivity = engine.app->activity->clazz;
  1636. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1637. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1638. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1639. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1640. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1641. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1642. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1643. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "CheckAlert", "()I");
  1644. jint jret = lJNIEnv->CallStaticIntMethod(T2DUtilitiesClass, MethodT2DUtilities);
  1645. int ret = jret;
  1646. lJNIEnv->DeleteLocalRef(strClassName);
  1647. // Finished with the JVM.
  1648. lJavaVM->DetachCurrentThread();
  1649. return ret;
  1650. }
  1651. void android_AlertOKCancel(const char *title, const char *message)
  1652. {
  1653. // Attaches the current thread to the JVM.
  1654. jint lResult;
  1655. jint lFlags = 0;
  1656. JavaVM* lJavaVM = engine.app->activity->vm;
  1657. JNIEnv* lJNIEnv = engine.app->activity->env;
  1658. JavaVMAttachArgs lJavaVMAttachArgs;
  1659. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1660. lJavaVMAttachArgs.name = "NativeThread";
  1661. lJavaVMAttachArgs.group = NULL;
  1662. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1663. if (lResult == JNI_ERR) {
  1664. return;
  1665. }
  1666. // Retrieves NativeActivity.
  1667. jobject lNativeActivity = engine.app->activity->clazz;
  1668. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1669. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1670. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1671. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1672. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1673. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1674. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1675. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1676. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1677. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertOKCancel", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1678. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1679. lJNIEnv->DeleteLocalRef(strClassName);
  1680. lJNIEnv->DeleteLocalRef(strTitle);
  1681. lJNIEnv->DeleteLocalRef(strMessage);
  1682. // Finished with the JVM.
  1683. lJavaVM->DetachCurrentThread();
  1684. }
  1685. void android_AlertRetry(const char *title, const char *message)
  1686. {
  1687. // Attaches the current thread to the JVM.
  1688. jint lResult;
  1689. jint lFlags = 0;
  1690. JavaVM* lJavaVM = engine.app->activity->vm;
  1691. JNIEnv* lJNIEnv = engine.app->activity->env;
  1692. JavaVMAttachArgs lJavaVMAttachArgs;
  1693. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1694. lJavaVMAttachArgs.name = "NativeThread";
  1695. lJavaVMAttachArgs.group = NULL;
  1696. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1697. if (lResult == JNI_ERR) {
  1698. return;
  1699. }
  1700. // Retrieves NativeActivity.
  1701. jobject lNativeActivity = engine.app->activity->clazz;
  1702. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1703. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1704. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1705. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1706. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1707. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1708. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1709. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1710. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1711. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertRetry", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1712. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1713. lJNIEnv->DeleteLocalRef(strClassName);
  1714. lJNIEnv->DeleteLocalRef(strTitle);
  1715. lJNIEnv->DeleteLocalRef(strMessage);
  1716. // Finished with the JVM.
  1717. lJavaVM->DetachCurrentThread();
  1718. }
  1719. void android_AlertYesNo(const char *title, const char *message)
  1720. {
  1721. // Attaches the current thread to the JVM.
  1722. jint lResult;
  1723. jint lFlags = 0;
  1724. JavaVM* lJavaVM = engine.app->activity->vm;
  1725. JNIEnv* lJNIEnv = engine.app->activity->env;
  1726. JavaVMAttachArgs lJavaVMAttachArgs;
  1727. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1728. lJavaVMAttachArgs.name = "NativeThread";
  1729. lJavaVMAttachArgs.group = NULL;
  1730. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1731. if (lResult == JNI_ERR) {
  1732. return;
  1733. }
  1734. // Retrieves NativeActivity.
  1735. jobject lNativeActivity = engine.app->activity->clazz;
  1736. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1737. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1738. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1739. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1740. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1741. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/T2DUtilities");
  1742. jclass T2DUtilitiesClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1743. jstring strTitle = lJNIEnv->NewStringUTF(title);
  1744. jstring strMessage = lJNIEnv->NewStringUTF(message);
  1745. jmethodID MethodT2DUtilities = lJNIEnv->GetStaticMethodID(T2DUtilitiesClass, "DisplayAlertYesNo", "(Landroid/content/Context;Ljava/lang/String;Ljava/lang/String;)V");
  1746. lJNIEnv->CallStaticVoidMethod(T2DUtilitiesClass, MethodT2DUtilities, lNativeActivity, strTitle, strMessage);
  1747. lJNIEnv->DeleteLocalRef(strClassName);
  1748. lJNIEnv->DeleteLocalRef(strTitle);
  1749. lJNIEnv->DeleteLocalRef(strMessage);
  1750. // Finished with the JVM.
  1751. lJavaVM->DetachCurrentThread();
  1752. }
  1753. void android_LoadMusicTrack( const char *mFilename )
  1754. {
  1755. // Attaches the current thread to the JVM.
  1756. jint lResult;
  1757. jint lFlags = 0;
  1758. JavaVM* lJavaVM = engine.app->activity->vm;
  1759. JNIEnv* lJNIEnv = engine.app->activity->env;
  1760. JavaVMAttachArgs lJavaVMAttachArgs;
  1761. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1762. lJavaVMAttachArgs.name = "NativeThread";
  1763. lJavaVMAttachArgs.group = NULL;
  1764. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1765. if (lResult == JNI_ERR) {
  1766. return;
  1767. }
  1768. // Retrieves NativeActivity.
  1769. jobject lNativeActivity = engine.app->activity->clazz;
  1770. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1771. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1772. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1773. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1774. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1775. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1776. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1777. jstring strFilename = lJNIEnv->NewStringUTF(mFilename);
  1778. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "LoadMusicTrack", "(Landroid/content/Context;Ljava/lang/String;)V");
  1779. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer, lNativeActivity, strFilename);
  1780. lJNIEnv->DeleteLocalRef(strClassName);
  1781. lJNIEnv->DeleteLocalRef(strFilename);
  1782. // Finished with the JVM.
  1783. lJavaVM->DetachCurrentThread();
  1784. }
  1785. void android_UnLoadMusicTrack()
  1786. {
  1787. // Attaches the current thread to the JVM.
  1788. jint lResult;
  1789. jint lFlags = 0;
  1790. JavaVM* lJavaVM = engine.app->activity->vm;
  1791. JNIEnv* lJNIEnv = engine.app->activity->env;
  1792. JavaVMAttachArgs lJavaVMAttachArgs;
  1793. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1794. lJavaVMAttachArgs.name = "NativeThread";
  1795. lJavaVMAttachArgs.group = NULL;
  1796. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1797. if (lResult == JNI_ERR) {
  1798. return;
  1799. }
  1800. // Retrieves NativeActivity.
  1801. jobject lNativeActivity = engine.app->activity->clazz;
  1802. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1803. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1804. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1805. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1806. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1807. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1808. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1809. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "UnLoadMusicTrack", "()V");
  1810. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1811. lJNIEnv->DeleteLocalRef(strClassName);
  1812. // Finished with the JVM.
  1813. lJavaVM->DetachCurrentThread();
  1814. }
  1815. bool android_isMusicTrackPlaying()
  1816. {
  1817. // Attaches the current thread to the JVM.
  1818. jint lResult;
  1819. jint lFlags = 0;
  1820. JavaVM* lJavaVM = engine.app->activity->vm;
  1821. JNIEnv* lJNIEnv = engine.app->activity->env;
  1822. JavaVMAttachArgs lJavaVMAttachArgs;
  1823. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1824. lJavaVMAttachArgs.name = "NativeThread";
  1825. lJavaVMAttachArgs.group = NULL;
  1826. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1827. if (lResult == JNI_ERR) {
  1828. return false;
  1829. }
  1830. // Retrieves NativeActivity.
  1831. jobject lNativeActivity = engine.app->activity->clazz;
  1832. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1833. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1834. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1835. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1836. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1837. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1838. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1839. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "isMusicTrackPlaying", "()Z");
  1840. jboolean jret = lJNIEnv->CallStaticBooleanMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1841. lJNIEnv->DeleteLocalRef(strClassName);
  1842. bool ret = jret;
  1843. // Finished with the JVM.
  1844. lJavaVM->DetachCurrentThread();
  1845. return ret;
  1846. }
  1847. void android_StartMusicTrack()
  1848. {
  1849. // Attaches the current thread to the JVM.
  1850. jint lResult;
  1851. jint lFlags = 0;
  1852. JavaVM* lJavaVM = engine.app->activity->vm;
  1853. JNIEnv* lJNIEnv = engine.app->activity->env;
  1854. JavaVMAttachArgs lJavaVMAttachArgs;
  1855. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1856. lJavaVMAttachArgs.name = "NativeThread";
  1857. lJavaVMAttachArgs.group = NULL;
  1858. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1859. if (lResult == JNI_ERR) {
  1860. return;
  1861. }
  1862. // Retrieves NativeActivity.
  1863. jobject lNativeActivity = engine.app->activity->clazz;
  1864. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1865. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1866. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1867. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1868. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1869. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1870. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1871. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "startMusicTrack", "()V");
  1872. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1873. lJNIEnv->DeleteLocalRef(strClassName);
  1874. // Finished with the JVM.
  1875. lJavaVM->DetachCurrentThread();
  1876. }
  1877. void android_StopMusicTrack()
  1878. {
  1879. // Attaches the current thread to the JVM.
  1880. jint lResult;
  1881. jint lFlags = 0;
  1882. JavaVM* lJavaVM = engine.app->activity->vm;
  1883. JNIEnv* lJNIEnv = engine.app->activity->env;
  1884. JavaVMAttachArgs lJavaVMAttachArgs;
  1885. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1886. lJavaVMAttachArgs.name = "NativeThread";
  1887. lJavaVMAttachArgs.group = NULL;
  1888. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1889. if (lResult == JNI_ERR) {
  1890. return;
  1891. }
  1892. // Retrieves NativeActivity.
  1893. jobject lNativeActivity = engine.app->activity->clazz;
  1894. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1895. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1896. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1897. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1898. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1899. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1900. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1901. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "stopMusicTrack", "()V");
  1902. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer);
  1903. lJNIEnv->DeleteLocalRef(strClassName);
  1904. // Finished with the JVM.
  1905. lJavaVM->DetachCurrentThread();
  1906. }
  1907. void android_setMusicTrackVolume(F32 volume)
  1908. {
  1909. // Attaches the current thread to the JVM.
  1910. jint lResult;
  1911. jint lFlags = 0;
  1912. JavaVM* lJavaVM = engine.app->activity->vm;
  1913. JNIEnv* lJNIEnv = engine.app->activity->env;
  1914. JavaVMAttachArgs lJavaVMAttachArgs;
  1915. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1916. lJavaVMAttachArgs.name = "NativeThread";
  1917. lJavaVMAttachArgs.group = NULL;
  1918. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1919. if (lResult == JNI_ERR) {
  1920. return;
  1921. }
  1922. // Retrieves NativeActivity.
  1923. jobject lNativeActivity = engine.app->activity->clazz;
  1924. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1925. jmethodID getClassLoader = lJNIEnv->GetMethodID(ClassNativeActivity,"getClassLoader", "()Ljava/lang/ClassLoader;");
  1926. jobject cls = lJNIEnv->CallObjectMethod(lNativeActivity, getClassLoader);
  1927. jclass classLoader = lJNIEnv->FindClass("java/lang/ClassLoader");
  1928. jmethodID findClass = lJNIEnv->GetMethodID(classLoader, "loadClass", "(Ljava/lang/String;)Ljava/lang/Class;");
  1929. jstring strClassName = lJNIEnv->NewStringUTF("com/garagegames/torque2d/StreamingAudioPlayer");
  1930. jclass StreamingAudioPlayerClass = (jclass)lJNIEnv->CallObjectMethod(cls, findClass, strClassName);
  1931. jmethodID MethodStreamingAudioPlayer = lJNIEnv->GetStaticMethodID(StreamingAudioPlayerClass, "setMusicTrackVolume", "(F)V");
  1932. lJNIEnv->CallStaticVoidMethod(StreamingAudioPlayerClass, MethodStreamingAudioPlayer, (jfloat)volume);
  1933. lJNIEnv->DeleteLocalRef(strClassName);
  1934. // Finished with the JVM.
  1935. lJavaVM->DetachCurrentThread();
  1936. }
  1937. ConsoleFunction(doDeviceVibrate, void, 1, 1, "Makes the device do a quick vibration. Only works on devices with vibration functionality.")
  1938. {
  1939. // Vibrate for 500 milliseconds
  1940. long vibrateTimeMS = 500L;
  1941. // Attaches the current thread to the JVM.
  1942. jint lResult;
  1943. jint lFlags = 0;
  1944. JavaVM* lJavaVM = platState.engine->app->activity->vm;
  1945. JNIEnv* lJNIEnv = platState.engine->app->activity->env;
  1946. JavaVMAttachArgs lJavaVMAttachArgs;
  1947. lJavaVMAttachArgs.version = JNI_VERSION_1_6;
  1948. lJavaVMAttachArgs.name = "NativeThread";
  1949. lJavaVMAttachArgs.group = NULL;
  1950. lResult=lJavaVM->AttachCurrentThread(&lJNIEnv, &lJavaVMAttachArgs);
  1951. if (lResult == JNI_ERR) {
  1952. return;
  1953. }
  1954. // Retrieves NativeActivity.
  1955. jobject lNativeActivity = platState.engine->app->activity->clazz;
  1956. jclass ClassNativeActivity = lJNIEnv->GetObjectClass(lNativeActivity);
  1957. // Retrieves Context.VIBRATOR_SERVICE.
  1958. jclass ClassContext = lJNIEnv->FindClass("android/content/Context");
  1959. jfieldID FieldVIBRATOR_SERVICE =lJNIEnv->GetStaticFieldID(ClassContext,"VIBRATOR_SERVICE", "Ljava/lang/String;");
  1960. jobject VIBRATOR_SERVICE = lJNIEnv->GetStaticObjectField(ClassContext, FieldVIBRATOR_SERVICE);
  1961. // Runs getSystemService(VIBRATOR_SERVICE)
  1962. jmethodID MethodGetSystemService = lJNIEnv->GetMethodID(ClassNativeActivity, "getSystemService","(Ljava/lang/String;)Ljava/lang/Object;");
  1963. jobject lVibrator = lJNIEnv->CallObjectMethod(lNativeActivity, MethodGetSystemService, VIBRATOR_SERVICE);
  1964. //Runs v.vibrate(ms)
  1965. jclass ClassVibrator = lJNIEnv->FindClass("android/os/Vibrator");
  1966. jmethodID MethodVibrate = lJNIEnv->GetMethodID(ClassVibrator, "vibrate", "(J)V");
  1967. lJNIEnv->CallVoidMethod(lVibrator, MethodVibrate, vibrateTimeMS);
  1968. // Finished with the JVM.
  1969. lJavaVM->DetachCurrentThread();
  1970. }
  1971. ConsoleFunction(enableAccelerometer, void, 1, 1, "() Allow accelerometer tracking during device motion updates")
  1972. {
  1973. activity.enableAccelerometer(true);
  1974. }
  1975. ConsoleFunction(disableAccelerometer, void, 1, 1, "() Stop accelerometer tracking")
  1976. {
  1977. activity.enableAccelerometer(false);
  1978. }
  1979. ConsoleFunction(isAccelerometerActive, bool, 1, 1, "() Check to see if Accelerometer is being polled\n"
  1980. "@return True if accelerometer is on, false otherwise")
  1981. {
  1982. return activity.isAccelerometerActive();
  1983. }
  1984. void adprintf(const char* fmt,...) {
  1985. va_list argptr;
  1986. int cnt;
  1987. char s[4096];
  1988. time_t now;
  1989. va_start(argptr,fmt);
  1990. cnt = vsprintf(s,fmt,argptr);
  1991. va_end(argptr);
  1992. time(&now);
  1993. tm* t = localtime(&now);
  1994. std::stringstream ss;
  1995. ss.clear();
  1996. ss << "[";
  1997. ss << (t->tm_year + 1900);
  1998. ss << "/";
  1999. if (t->tm_mon < 9)
  2000. ss << "0";
  2001. ss << (t->tm_mon+1);
  2002. ss << "/";
  2003. if (t->tm_mday < 10)
  2004. ss << "0";
  2005. ss << t->tm_mday;
  2006. ss << " ";
  2007. if (t->tm_hour < 10)
  2008. ss << "0";
  2009. ss << t->tm_hour;
  2010. ss << ":";
  2011. if (t->tm_min < 10)
  2012. ss << "0";
  2013. ss << t->tm_min;
  2014. ss << ":";
  2015. if (t->tm_sec < 10)
  2016. ss << "0";
  2017. ss << t->tm_sec;
  2018. ss << "] ";
  2019. ss << s;
  2020. __android_log_print(ANDROID_LOG_INFO, "Torque2D", "%s", ss.str().c_str());
  2021. }