| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2026 Sam Lantinga <slouken@libsdl.org>
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any damages
- arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it
- freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not
- claim that you wrote the original software. If you use this software
- in a product, an acknowledgment in the product documentation would be
- appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be
- misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include "SDL_internal.h"
- #ifdef SDL_PLATFORM_ANDROID
- #include "SDL_android.h"
- #include "../../events/SDL_events_c.h"
- #include "../../video/android/SDL_androidkeyboard.h"
- #include "../../video/android/SDL_androidmouse.h"
- #include "../../video/android/SDL_androidtouch.h"
- #include "../../video/android/SDL_androidpen.h"
- #include "../../video/android/SDL_androidvideo.h"
- #include "../../video/android/SDL_androidwindow.h"
- #include "../../joystick/android/SDL_sysjoystick_c.h"
- #include "../../haptic/android/SDL_syshaptic_c.h"
- #include "../../hidapi/android/hid.h"
- #include "../../SDL_hints_c.h"
- #include <android/log.h>
- #include <android/configuration.h>
- #include <android/asset_manager_jni.h>
- #include <sys/system_properties.h>
- #include <pthread.h>
- #include <sys/types.h>
- #include <unistd.h>
- #include <dlfcn.h>
- #include <time.h>
- #define SDL_JAVA_PREFIX org_libsdl_app
- #define CONCAT1(prefix, class, function) CONCAT2(prefix, class, function)
- #define CONCAT2(prefix, class, function) Java_##prefix##_##class##_##function
- #define SDL_JAVA_INTERFACE(function) CONCAT1(SDL_JAVA_PREFIX, SDLActivity, function)
- #define SDL_JAVA_AUDIO_INTERFACE(function) CONCAT1(SDL_JAVA_PREFIX, SDLAudioManager, function)
- #define SDL_JAVA_CONTROLLER_INTERFACE(function) CONCAT1(SDL_JAVA_PREFIX, SDLControllerManager, function)
- #define SDL_JAVA_INTERFACE_INPUT_CONNECTION(function) CONCAT1(SDL_JAVA_PREFIX, SDLInputConnection, function)
- // Audio encoding definitions
- #define ENCODING_PCM_8BIT 3
- #define ENCODING_PCM_16BIT 2
- #define ENCODING_PCM_FLOAT 4
- // Java class SDLActivity
- JNIEXPORT jstring JNICALL SDL_JAVA_INTERFACE(nativeGetVersion)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetupJNI)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeInitMainThread)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeCleanupMainThread)(
- JNIEnv *env, jclass cls);
- JNIEXPORT int JNICALL SDL_JAVA_INTERFACE(nativeRunMain)(
- JNIEnv *env, jclass cls,
- jstring library, jstring function, jobject array);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeDropFile)(
- JNIEnv *env, jclass jcls,
- jstring filename);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetScreenResolution)(
- JNIEnv *env, jclass jcls,
- jint surfaceWidth, jint surfaceHeight,
- jint deviceWidth, jint deviceHeight, jfloat density, jfloat rate);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeResize)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceCreated)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceChanged)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceDestroyed)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeScreenKeyboardShown)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeScreenKeyboardHidden)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyDown)(
- JNIEnv *env, jclass jcls,
- jint keycode);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyUp)(
- JNIEnv *env, jclass jcls,
- jint keycode);
- JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(onNativeSoftReturnKey)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyboardFocusLost)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeTouch)(
- JNIEnv *env, jclass jcls,
- jint touch_device_id_in, jint pointer_finger_id_in,
- jint action, jfloat x, jfloat y, jfloat p);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePinchStart)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePinchUpdate)(
- JNIEnv *env, jclass jcls,
- jfloat scale);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePinchEnd)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeMouse)(
- JNIEnv *env, jclass jcls,
- jint button, jint action, jfloat x, jfloat y, jboolean relative);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePen)(
- JNIEnv *env, jclass jcls,
- jint pen_id_in, jint device_type, jint button, jint action, jfloat x, jfloat y, jfloat p);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeAccel)(
- JNIEnv *env, jclass jcls,
- jfloat x, jfloat y, jfloat z);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeClipboardChanged)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeLowMemory)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeLocaleChanged)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeDarkModeChanged)(
- JNIEnv *env, jclass cls, jboolean enabled);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSendQuit)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeQuit)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePause)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeResume)(
- JNIEnv *env, jclass cls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeFocusChanged)(
- JNIEnv *env, jclass cls, jboolean hasFocus);
- JNIEXPORT jstring JNICALL SDL_JAVA_INTERFACE(nativeGetHint)(
- JNIEnv *env, jclass cls,
- jstring name);
- JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(nativeGetHintBoolean)(
- JNIEnv *env, jclass cls,
- jstring name, jboolean default_value);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetenv)(
- JNIEnv *env, jclass cls,
- jstring name, jstring value);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetNaturalOrientation)(
- JNIEnv *env, jclass cls,
- jint orientation);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeRotationChanged)(
- JNIEnv *env, jclass cls,
- jint rotation);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeInsetsChanged)(
- JNIEnv *env, jclass cls,
- jint left, jint right, jint top, jint bottom);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeAddTouch)(
- JNIEnv *env, jclass cls,
- jint touchId, jstring name);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePermissionResult)(
- JNIEnv *env, jclass cls,
- jint requestCode, jboolean result);
- JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(nativeAllowRecreateActivity)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT int JNICALL SDL_JAVA_INTERFACE(nativeCheckSDLThreadCounter)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeFileDialog)(
- JNIEnv *env, jclass jcls,
- jint requestCode, jobjectArray fileList, jint filter);
- static JNINativeMethod SDLActivity_tab[] = {
- { "nativeGetVersion", "()Ljava/lang/String;", SDL_JAVA_INTERFACE(nativeGetVersion) },
- { "nativeSetupJNI", "()V", SDL_JAVA_INTERFACE(nativeSetupJNI) },
- { "nativeInitMainThread", "()V", SDL_JAVA_INTERFACE(nativeInitMainThread) },
- { "nativeCleanupMainThread", "()V", SDL_JAVA_INTERFACE(nativeCleanupMainThread) },
- { "nativeRunMain", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/Object;)I", SDL_JAVA_INTERFACE(nativeRunMain) },
- { "onNativeDropFile", "(Ljava/lang/String;)V", SDL_JAVA_INTERFACE(onNativeDropFile) },
- { "nativeSetScreenResolution", "(IIIIFF)V", SDL_JAVA_INTERFACE(nativeSetScreenResolution) },
- { "onNativeResize", "()V", SDL_JAVA_INTERFACE(onNativeResize) },
- { "onNativeSurfaceCreated", "()V", SDL_JAVA_INTERFACE(onNativeSurfaceCreated) },
- { "onNativeSurfaceChanged", "()V", SDL_JAVA_INTERFACE(onNativeSurfaceChanged) },
- { "onNativeSurfaceDestroyed", "()V", SDL_JAVA_INTERFACE(onNativeSurfaceDestroyed) },
- { "onNativeScreenKeyboardShown", "()V", SDL_JAVA_INTERFACE(onNativeScreenKeyboardShown) },
- { "onNativeScreenKeyboardHidden", "()V", SDL_JAVA_INTERFACE(onNativeScreenKeyboardHidden) },
- { "onNativeKeyDown", "(I)V", SDL_JAVA_INTERFACE(onNativeKeyDown) },
- { "onNativeKeyUp", "(I)V", SDL_JAVA_INTERFACE(onNativeKeyUp) },
- { "onNativeSoftReturnKey", "()Z", SDL_JAVA_INTERFACE(onNativeSoftReturnKey) },
- { "onNativeKeyboardFocusLost", "()V", SDL_JAVA_INTERFACE(onNativeKeyboardFocusLost) },
- { "onNativeTouch", "(IIIFFF)V", SDL_JAVA_INTERFACE(onNativeTouch) },
- { "onNativePinchStart", "()V", SDL_JAVA_INTERFACE(onNativePinchStart) },
- { "onNativePinchUpdate", "(F)V", SDL_JAVA_INTERFACE(onNativePinchUpdate) },
- { "onNativePinchEnd", "()V", SDL_JAVA_INTERFACE(onNativePinchEnd) },
- { "onNativeMouse", "(IIFFZ)V", SDL_JAVA_INTERFACE(onNativeMouse) },
- { "onNativePen", "(IIIIFFF)V", SDL_JAVA_INTERFACE(onNativePen) },
- { "onNativeAccel", "(FFF)V", SDL_JAVA_INTERFACE(onNativeAccel) },
- { "onNativeClipboardChanged", "()V", SDL_JAVA_INTERFACE(onNativeClipboardChanged) },
- { "nativeLowMemory", "()V", SDL_JAVA_INTERFACE(nativeLowMemory) },
- { "onNativeLocaleChanged", "()V", SDL_JAVA_INTERFACE(onNativeLocaleChanged) },
- { "onNativeDarkModeChanged", "(Z)V", SDL_JAVA_INTERFACE(onNativeDarkModeChanged) },
- { "nativeSendQuit", "()V", SDL_JAVA_INTERFACE(nativeSendQuit) },
- { "nativeQuit", "()V", SDL_JAVA_INTERFACE(nativeQuit) },
- { "nativePause", "()V", SDL_JAVA_INTERFACE(nativePause) },
- { "nativeResume", "()V", SDL_JAVA_INTERFACE(nativeResume) },
- { "nativeFocusChanged", "(Z)V", SDL_JAVA_INTERFACE(nativeFocusChanged) },
- { "nativeGetHint", "(Ljava/lang/String;)Ljava/lang/String;", SDL_JAVA_INTERFACE(nativeGetHint) },
- { "nativeGetHintBoolean", "(Ljava/lang/String;Z)Z", SDL_JAVA_INTERFACE(nativeGetHintBoolean) },
- { "nativeSetenv", "(Ljava/lang/String;Ljava/lang/String;)V", SDL_JAVA_INTERFACE(nativeSetenv) },
- { "nativeSetNaturalOrientation", "(I)V", SDL_JAVA_INTERFACE(nativeSetNaturalOrientation) },
- { "onNativeRotationChanged", "(I)V", SDL_JAVA_INTERFACE(onNativeRotationChanged) },
- { "onNativeInsetsChanged", "(IIII)V", SDL_JAVA_INTERFACE(onNativeInsetsChanged) },
- { "nativeAddTouch", "(ILjava/lang/String;)V", SDL_JAVA_INTERFACE(nativeAddTouch) },
- { "nativePermissionResult", "(IZ)V", SDL_JAVA_INTERFACE(nativePermissionResult) },
- { "nativeAllowRecreateActivity", "()Z", SDL_JAVA_INTERFACE(nativeAllowRecreateActivity) },
- { "nativeCheckSDLThreadCounter", "()I", SDL_JAVA_INTERFACE(nativeCheckSDLThreadCounter) },
- { "onNativeFileDialog", "(I[Ljava/lang/String;I)V", SDL_JAVA_INTERFACE(onNativeFileDialog) }
- };
- // Java class SDLInputConnection
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeCommitText)(
- JNIEnv *env, jclass cls,
- jstring text, jint newCursorPosition);
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeGenerateScancodeForUnichar)(
- JNIEnv *env, jclass cls,
- jchar chUnicode);
- static JNINativeMethod SDLInputConnection_tab[] = {
- { "nativeCommitText", "(Ljava/lang/String;I)V", SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeCommitText) },
- { "nativeGenerateScancodeForUnichar", "(C)V", SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeGenerateScancodeForUnichar) }
- };
- // Java class SDLAudioManager
- JNIEXPORT void JNICALL SDL_JAVA_AUDIO_INTERFACE(nativeSetupJNI)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT void JNICALL
- SDL_JAVA_AUDIO_INTERFACE(nativeAddAudioDevice)(JNIEnv *env, jclass jcls, jboolean recording, jstring name,
- jint device_id);
- JNIEXPORT void JNICALL
- SDL_JAVA_AUDIO_INTERFACE(nativeRemoveAudioDevice)(JNIEnv *env, jclass jcls, jboolean recording,
- jint device_id);
- static JNINativeMethod SDLAudioManager_tab[] = {
- { "nativeSetupJNI", "()V", SDL_JAVA_AUDIO_INTERFACE(nativeSetupJNI) },
- { "nativeAddAudioDevice", "(ZLjava/lang/String;I)V", SDL_JAVA_AUDIO_INTERFACE(nativeAddAudioDevice) },
- { "nativeRemoveAudioDevice", "(ZI)V", SDL_JAVA_AUDIO_INTERFACE(nativeRemoveAudioDevice) }
- };
- // Java class SDLControllerManager
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeSetupJNI)(
- JNIEnv *env, jclass jcls);
- JNIEXPORT jboolean JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadDown)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint keycode);
- JNIEXPORT jboolean JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadUp)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint keycode);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoy)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint axis, jfloat value);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeHat)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint hat_id, jint x, jint y);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoySensor)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint sensor_type, jlong sensor_timestamp, jfloat x, jfloat y, jfloat z);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddJoystick)(
- JNIEnv *env, jclass jcls,
- jint device_id, jstring device_name, jstring device_desc, jint vendor_id, jint product_id,
- jint button_mask, jint naxes, jint axis_mask, jint nhats,
- jboolean can_rumble, jboolean has_rgb_led, jboolean has_accelerometer, jboolean has_gyroscope);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveJoystick)(
- JNIEnv *env, jclass jcls,
- jint device_id);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddHaptic)(
- JNIEnv *env, jclass jcls,
- jint device_id, jstring device_name);
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveHaptic)(
- JNIEnv *env, jclass jcls,
- jint device_id);
- static JNINativeMethod SDLControllerManager_tab[] = {
- { "nativeSetupJNI", "()V", SDL_JAVA_CONTROLLER_INTERFACE(nativeSetupJNI) },
- { "onNativePadDown", "(II)Z", SDL_JAVA_CONTROLLER_INTERFACE(onNativePadDown) },
- { "onNativePadUp", "(II)Z", SDL_JAVA_CONTROLLER_INTERFACE(onNativePadUp) },
- { "onNativeJoy", "(IIF)V", SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoy) },
- { "onNativeHat", "(IIII)V", SDL_JAVA_CONTROLLER_INTERFACE(onNativeHat) },
- { "onNativeJoySensor", "(IIJFFF)V", SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoySensor) },
- { "nativeAddJoystick", "(ILjava/lang/String;Ljava/lang/String;IIIIIIZZZZ)V", SDL_JAVA_CONTROLLER_INTERFACE(nativeAddJoystick) },
- { "nativeRemoveJoystick", "(I)V", SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveJoystick) },
- { "nativeAddHaptic", "(ILjava/lang/String;)V", SDL_JAVA_CONTROLLER_INTERFACE(nativeAddHaptic) },
- { "nativeRemoveHaptic", "(I)V", SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveHaptic) }
- };
- // Uncomment this to log messages entering and exiting methods in this file
- // #define DEBUG_JNI
- static void checkJNIReady(void);
- /*******************************************************************************
- This file links the Java side of Android with libsdl
- *******************************************************************************/
- #include <jni.h>
- /*******************************************************************************
- Globals
- *******************************************************************************/
- static pthread_key_t mThreadKey;
- static pthread_once_t key_once = PTHREAD_ONCE_INIT;
- static JavaVM *mJavaVM = NULL;
- // Main activity
- static jclass mActivityClass;
- // method signatures
- static jmethodID midClipboardGetText;
- static jmethodID midClipboardHasText;
- static jmethodID midClipboardSetText;
- static jmethodID midCreateCustomCursor;
- static jmethodID midDestroyCustomCursor;
- static jmethodID midGetContext;
- static jmethodID midGetManifestEnvironmentVariables;
- static jmethodID midGetNativeSurface;
- static jmethodID midInitTouch;
- static jmethodID midIsAndroidTV;
- static jmethodID midIsChromebook;
- static jmethodID midIsDeXMode;
- static jmethodID midIsTablet;
- static jmethodID midManualBackButton;
- static jmethodID midMinimizeWindow;
- static jmethodID midOpenURL;
- static jmethodID midRequestPermission;
- static jmethodID midShowToast;
- static jmethodID midSendMessage;
- static jmethodID midSetActivityTitle;
- static jmethodID midSetCustomCursor;
- static jmethodID midSetOrientation;
- static jmethodID midSetRelativeMouseEnabled;
- static jmethodID midSetSystemCursor;
- static jmethodID midSetWindowStyle;
- static jmethodID midShouldMinimizeOnFocusLoss;
- static jmethodID midShowTextInput;
- static jmethodID midSupportsRelativeMouse;
- static jmethodID midOpenFileDescriptor;
- static jmethodID midShowFileDialog;
- static jmethodID midGetPreferredLocales;
- // audio manager
- static jclass mAudioManagerClass;
- // method signatures
- static jmethodID midRegisterAudioDeviceCallback;
- static jmethodID midUnregisterAudioDeviceCallback;
- static jmethodID midAudioSetThreadPriority;
- // controller manager
- static jclass mControllerManagerClass;
- // method signatures
- static jmethodID midPollInputDevices;
- static jmethodID midJoystickSetLED;
- static jmethodID midJoystickSetSensorsEnabled;
- static jmethodID midPollHapticDevices;
- static jmethodID midHapticRun;
- static jmethodID midHapticRumble;
- static jmethodID midHapticStop;
- // Accelerometer data storage
- static SDL_DisplayOrientation displayNaturalOrientation;
- static SDL_DisplayOrientation displayCurrentOrientation;
- static float fLastAccelerometer[3];
- static bool bHasNewData;
- static bool bHasEnvironmentVariables;
- // Android AssetManager
- static void Internal_Android_Create_AssetManager(void);
- static void Internal_Android_Destroy_AssetManager(void);
- static AAssetManager *asset_manager = NULL;
- static jobject javaAssetManagerRef = 0;
- static SDL_Mutex *Android_ActivityMutex = NULL;
- static SDL_Mutex *Android_LifecycleMutex = NULL;
- static SDL_Semaphore *Android_LifecycleEventSem = NULL;
- static SDL_AndroidLifecycleEvent Android_LifecycleEvents[SDL_NUM_ANDROID_LIFECYCLE_EVENTS];
- static int Android_NumLifecycleEvents;
- /*******************************************************************************
- Functions called by JNI
- *******************************************************************************/
- /* From http://developer.android.com/guide/practices/jni.html
- * All threads are Linux threads, scheduled by the kernel.
- * They're usually started from managed code (using Thread.start), but they can also be created elsewhere and then
- * attached to the JavaVM. For example, a thread started with pthread_create can be attached with the
- * JNI AttachCurrentThread or AttachCurrentThreadAsDaemon functions. Until a thread is attached, it has no JNIEnv,
- * and cannot make JNI calls.
- * Attaching a natively-created thread causes a java.lang.Thread object to be constructed and added to the "main"
- * ThreadGroup, making it visible to the debugger. Calling AttachCurrentThread on an already-attached thread
- * is a no-op.
- * Note: You can call this function any number of times for the same thread, there's no harm in it
- */
- /* From http://developer.android.com/guide/practices/jni.html
- * Threads attached through JNI must call DetachCurrentThread before they exit. If coding this directly is awkward,
- * in Android 2.0 (Eclair) and higher you can use pthread_key_create to define a destructor function that will be
- * called before the thread exits, and call DetachCurrentThread from there. (Use that key with pthread_setspecific
- * to store the JNIEnv in thread-local-storage; that way it'll be passed into your destructor as the argument.)
- * Note: The destructor is not called unless the stored value is != NULL
- * Note: You can call this function any number of times for the same thread, there's no harm in it
- * (except for some lost CPU cycles)
- */
- // Set local storage value
- static bool Android_JNI_SetEnv(JNIEnv *env)
- {
- int status = pthread_setspecific(mThreadKey, env);
- if (status < 0) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed pthread_setspecific() in Android_JNI_SetEnv() (err=%d)", status);
- return false;
- }
- return true;
- }
- // Get local storage value
- JNIEnv *Android_JNI_GetEnv(void)
- {
- // Get JNIEnv from the Thread local storage
- JNIEnv *env = pthread_getspecific(mThreadKey);
- if (!env) {
- // If it fails, try to attach ! (e.g the thread isn't created with SDL_CreateThread()
- int status;
- // There should be a JVM
- if (!mJavaVM) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed, there is no JavaVM");
- return NULL;
- }
- /* Attach the current thread to the JVM and get a JNIEnv.
- * It will be detached by pthread_create destructor 'Android_JNI_ThreadDestroyed' */
- status = (*mJavaVM)->AttachCurrentThread(mJavaVM, &env, NULL);
- if (status < 0) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to attach current thread (err=%d)", status);
- return NULL;
- }
- // Save JNIEnv into the Thread local storage
- if (!Android_JNI_SetEnv(env)) {
- return NULL;
- }
- }
- return env;
- }
- // Set up an external thread for using JNI with Android_JNI_GetEnv()
- bool Android_JNI_SetupThread(void)
- {
- JNIEnv *env;
- int status;
- // There should be a JVM
- if (!mJavaVM) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed, there is no JavaVM");
- return false;
- }
- /* Attach the current thread to the JVM and get a JNIEnv.
- * It will be detached by pthread_create destructor 'Android_JNI_ThreadDestroyed' */
- status = (*mJavaVM)->AttachCurrentThread(mJavaVM, &env, NULL);
- if (status < 0) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to attach current thread (err=%d)", status);
- return false;
- }
- // Save JNIEnv into the Thread local storage
- if (!Android_JNI_SetEnv(env)) {
- return false;
- }
- return true;
- }
- // Destructor called for each thread where mThreadKey is not NULL
- static void Android_JNI_ThreadDestroyed(void *value)
- {
- // The thread is being destroyed, detach it from the Java VM and set the mThreadKey value to NULL as required
- JNIEnv *env = (JNIEnv *)value;
- if (env) {
- (*mJavaVM)->DetachCurrentThread(mJavaVM);
- Android_JNI_SetEnv(NULL);
- }
- }
- // Creation of local storage mThreadKey
- static void Android_JNI_CreateKey(void)
- {
- int status = pthread_key_create(&mThreadKey, Android_JNI_ThreadDestroyed);
- if (status < 0) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Error initializing mThreadKey with pthread_key_create() (err=%d)", status);
- }
- }
- static void Android_JNI_CreateKey_once(void)
- {
- int status = pthread_once(&key_once, Android_JNI_CreateKey);
- if (status < 0) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Error initializing mThreadKey with pthread_once() (err=%d)", status);
- }
- }
- static void register_methods(JNIEnv *env, const char *classname, JNINativeMethod *methods, int nb)
- {
- jclass clazz = (*env)->FindClass(env, classname);
- if (!clazz || (*env)->RegisterNatives(env, clazz, methods, nb) < 0) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to register methods of %s", classname);
- return;
- }
- }
- // Library init
- JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved)
- {
- JNIEnv *env = NULL;
- mJavaVM = vm;
- if ((*mJavaVM)->GetEnv(mJavaVM, (void **)&env, JNI_VERSION_1_4) != JNI_OK) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "Failed to get JNI Env");
- return JNI_VERSION_1_4;
- }
- register_methods(env, "org/libsdl/app/SDLActivity", SDLActivity_tab, SDL_arraysize(SDLActivity_tab));
- register_methods(env, "org/libsdl/app/SDLInputConnection", SDLInputConnection_tab, SDL_arraysize(SDLInputConnection_tab));
- register_methods(env, "org/libsdl/app/SDLAudioManager", SDLAudioManager_tab, SDL_arraysize(SDLAudioManager_tab));
- register_methods(env, "org/libsdl/app/SDLControllerManager", SDLControllerManager_tab, SDL_arraysize(SDLControllerManager_tab));
- register_methods(env, "org/libsdl/app/HIDDeviceManager", HIDDeviceManager_tab, SDL_arraysize(HIDDeviceManager_tab));
- return JNI_VERSION_1_4;
- }
- void checkJNIReady(void)
- {
- if (!mActivityClass || !mAudioManagerClass || !mControllerManagerClass) {
- // We aren't fully initialized, let's just return.
- return;
- }
- SDL_SetMainReady();
- }
- // Get SDL version -- called before SDL_main() to verify JNI bindings
- JNIEXPORT jstring JNICALL SDL_JAVA_INTERFACE(nativeGetVersion)(JNIEnv *env, jclass cls)
- {
- char version[128];
- SDL_snprintf(version, sizeof(version), "%d.%d.%d", SDL_MAJOR_VERSION, SDL_MINOR_VERSION, SDL_MICRO_VERSION);
- return (*env)->NewStringUTF(env, version);
- }
- // Activity initialization -- called before SDL_main() to initialize JNI bindings
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetupJNI)(JNIEnv *env, jclass cls)
- {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeSetupJNI()");
- // Start with a clean slate
- SDL_ClearError();
- /*
- * Create mThreadKey so we can keep track of the JNIEnv assigned to each thread
- * Refer to http://developer.android.com/guide/practices/design/jni.html for the rationale behind this
- */
- Android_JNI_CreateKey_once();
- // Save JNIEnv of SDLActivity
- Android_JNI_SetEnv(env);
- if (!mJavaVM) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to found a JavaVM");
- }
- /* Use a mutex to prevent concurrency issues between Java Activity and Native thread code, when using 'Android_Window'.
- * (Eg. Java sending Touch events, while native code is destroying the main SDL_Window. )
- */
- if (!Android_ActivityMutex) {
- Android_ActivityMutex = SDL_CreateMutex(); // Could this be created twice if onCreate() is called a second time ?
- }
- if (!Android_ActivityMutex) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to create Android_ActivityMutex mutex");
- }
- Android_LifecycleMutex = SDL_CreateMutex();
- if (!Android_LifecycleMutex) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to create Android_LifecycleMutex mutex");
- }
- Android_LifecycleEventSem = SDL_CreateSemaphore(0);
- if (!Android_LifecycleEventSem) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "failed to create Android_LifecycleEventSem semaphore");
- }
- mActivityClass = (jclass)((*env)->NewGlobalRef(env, cls));
- midClipboardGetText = (*env)->GetStaticMethodID(env, mActivityClass, "clipboardGetText", "()Ljava/lang/String;");
- midClipboardHasText = (*env)->GetStaticMethodID(env, mActivityClass, "clipboardHasText", "()Z");
- midClipboardSetText = (*env)->GetStaticMethodID(env, mActivityClass, "clipboardSetText", "(Ljava/lang/String;)V");
- midCreateCustomCursor = (*env)->GetStaticMethodID(env, mActivityClass, "createCustomCursor", "([IIIII)I");
- midDestroyCustomCursor = (*env)->GetStaticMethodID(env, mActivityClass, "destroyCustomCursor", "(I)V");
- midGetContext = (*env)->GetStaticMethodID(env, mActivityClass, "getContext", "()Landroid/app/Activity;");
- midGetManifestEnvironmentVariables = (*env)->GetStaticMethodID(env, mActivityClass, "getManifestEnvironmentVariables", "()Z");
- midGetNativeSurface = (*env)->GetStaticMethodID(env, mActivityClass, "getNativeSurface", "()Landroid/view/Surface;");
- midInitTouch = (*env)->GetStaticMethodID(env, mActivityClass, "initTouch", "()V");
- midIsAndroidTV = (*env)->GetStaticMethodID(env, mActivityClass, "isAndroidTV", "()Z");
- midIsChromebook = (*env)->GetStaticMethodID(env, mActivityClass, "isChromebook", "()Z");
- midIsDeXMode = (*env)->GetStaticMethodID(env, mActivityClass, "isDeXMode", "()Z");
- midIsTablet = (*env)->GetStaticMethodID(env, mActivityClass, "isTablet", "()Z");
- midManualBackButton = (*env)->GetStaticMethodID(env, mActivityClass, "manualBackButton", "()V");
- midMinimizeWindow = (*env)->GetStaticMethodID(env, mActivityClass, "minimizeWindow", "()V");
- midOpenURL = (*env)->GetStaticMethodID(env, mActivityClass, "openURL", "(Ljava/lang/String;)Z");
- midRequestPermission = (*env)->GetStaticMethodID(env, mActivityClass, "requestPermission", "(Ljava/lang/String;I)V");
- midShowToast = (*env)->GetStaticMethodID(env, mActivityClass, "showToast", "(Ljava/lang/String;IIII)Z");
- midSendMessage = (*env)->GetStaticMethodID(env, mActivityClass, "sendMessage", "(II)Z");
- midSetActivityTitle = (*env)->GetStaticMethodID(env, mActivityClass, "setActivityTitle", "(Ljava/lang/String;)Z");
- midSetCustomCursor = (*env)->GetStaticMethodID(env, mActivityClass, "setCustomCursor", "(I)Z");
- midSetOrientation = (*env)->GetStaticMethodID(env, mActivityClass, "setOrientation", "(IIZLjava/lang/String;)V");
- midSetRelativeMouseEnabled = (*env)->GetStaticMethodID(env, mActivityClass, "setRelativeMouseEnabled", "(Z)Z");
- midSetSystemCursor = (*env)->GetStaticMethodID(env, mActivityClass, "setSystemCursor", "(I)Z");
- midSetWindowStyle = (*env)->GetStaticMethodID(env, mActivityClass, "setWindowStyle", "(Z)V");
- midShouldMinimizeOnFocusLoss = (*env)->GetStaticMethodID(env, mActivityClass, "shouldMinimizeOnFocusLoss", "()Z");
- midShowTextInput = (*env)->GetStaticMethodID(env, mActivityClass, "showTextInput", "(IIIII)Z");
- midSupportsRelativeMouse = (*env)->GetStaticMethodID(env, mActivityClass, "supportsRelativeMouse", "()Z");
- midOpenFileDescriptor = (*env)->GetStaticMethodID(env, mActivityClass, "openFileDescriptor", "(Ljava/lang/String;Ljava/lang/String;)I");
- midShowFileDialog = (*env)->GetStaticMethodID(env, mActivityClass, "showFileDialog", "([Ljava/lang/String;ZZI)Z");
- midGetPreferredLocales = (*env)->GetStaticMethodID(env, mActivityClass, "getPreferredLocales", "()Ljava/lang/String;");
- if (!midClipboardGetText ||
- !midClipboardHasText ||
- !midClipboardSetText ||
- !midCreateCustomCursor ||
- !midDestroyCustomCursor ||
- !midGetContext ||
- !midGetManifestEnvironmentVariables ||
- !midGetNativeSurface ||
- !midInitTouch ||
- !midIsAndroidTV ||
- !midIsChromebook ||
- !midIsDeXMode ||
- !midIsTablet ||
- !midManualBackButton ||
- !midMinimizeWindow ||
- !midOpenURL ||
- !midRequestPermission ||
- !midShowToast ||
- !midSendMessage ||
- !midSetActivityTitle ||
- !midSetCustomCursor ||
- !midSetOrientation ||
- !midSetRelativeMouseEnabled ||
- !midSetSystemCursor ||
- !midSetWindowStyle ||
- !midShouldMinimizeOnFocusLoss ||
- !midShowTextInput ||
- !midSupportsRelativeMouse ||
- !midOpenFileDescriptor ||
- !midShowFileDialog ||
- !midGetPreferredLocales) {
- __android_log_print(ANDROID_LOG_WARN, "SDL", "Missing some Java callbacks, do you have the latest version of SDLActivity.java?");
- }
- checkJNIReady();
- }
- // Audio initialization -- called before SDL_main() to initialize JNI bindings
- JNIEXPORT void JNICALL SDL_JAVA_AUDIO_INTERFACE(nativeSetupJNI)(JNIEnv *env, jclass cls)
- {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "AUDIO nativeSetupJNI()");
- mAudioManagerClass = (jclass)((*env)->NewGlobalRef(env, cls));
- midRegisterAudioDeviceCallback = (*env)->GetStaticMethodID(env, mAudioManagerClass,
- "registerAudioDeviceCallback",
- "()V");
- midUnregisterAudioDeviceCallback = (*env)->GetStaticMethodID(env, mAudioManagerClass,
- "unregisterAudioDeviceCallback",
- "()V");
- midAudioSetThreadPriority = (*env)->GetStaticMethodID(env, mAudioManagerClass,
- "audioSetThreadPriority", "(ZI)V");
- if (!midRegisterAudioDeviceCallback || !midUnregisterAudioDeviceCallback || !midAudioSetThreadPriority) {
- __android_log_print(ANDROID_LOG_WARN, "SDL",
- "Missing some Java callbacks, do you have the latest version of SDLAudioManager.java?");
- }
- checkJNIReady();
- }
- // Controller initialization -- called before SDL_main() to initialize JNI bindings
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeSetupJNI)(JNIEnv *env, jclass cls)
- {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "CONTROLLER nativeSetupJNI()");
- mControllerManagerClass = (jclass)((*env)->NewGlobalRef(env, cls));
- midPollInputDevices = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "pollInputDevices", "()V");
- midJoystickSetLED = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "joystickSetLED", "(IIII)V");
- midJoystickSetSensorsEnabled = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "joystickSetSensorsEnabled", "(IZ)V");
- midPollHapticDevices = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "pollHapticDevices", "()V");
- midHapticRun = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "hapticRun", "(IFI)V");
- midHapticRumble = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "hapticRumble", "(IFFI)V");
- midHapticStop = (*env)->GetStaticMethodID(env, mControllerManagerClass,
- "hapticStop", "(I)V");
- if (!midPollInputDevices || !midJoystickSetLED || !midJoystickSetSensorsEnabled || !midPollHapticDevices || !midHapticRun || !midHapticRumble || !midHapticStop) {
- __android_log_print(ANDROID_LOG_WARN, "SDL", "Missing some Java callbacks, do you have the latest version of SDLControllerManager.java?");
- }
- checkJNIReady();
- }
- static int run_count = 0;
- static bool allow_recreate_activity;
- static bool allow_recreate_activity_set;
- JNIEXPORT int JNICALL SDL_JAVA_INTERFACE(nativeCheckSDLThreadCounter)(
- JNIEnv *env, jclass jcls)
- {
- int tmp = run_count;
- run_count += 1;
- return tmp;
- }
- void Android_SetAllowRecreateActivity(bool enabled)
- {
- allow_recreate_activity = enabled;
- allow_recreate_activity_set = true;
- }
- JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(nativeAllowRecreateActivity)(
- JNIEnv *env, jclass jcls)
- {
- return allow_recreate_activity;
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeInitMainThread)(
- JNIEnv *env, jclass jcls)
- {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeInitSDLThread() %d time", run_count);
- run_count += 1;
- // Save JNIEnv of SDLThread
- Android_JNI_SetEnv(env);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeCleanupMainThread)(
- JNIEnv *env, jclass jcls)
- {
- /* This is a Java thread, it doesn't need to be Detached from the JVM.
- * Set to mThreadKey value to NULL not to call pthread_create destructor 'Android_JNI_ThreadDestroyed' */
- Android_JNI_SetEnv(NULL);
- }
- // Start up the SDL app
- JNIEXPORT int JNICALL SDL_JAVA_INTERFACE(nativeRunMain)(JNIEnv *env, jclass cls, jstring library, jstring function, jobject array)
- {
- int status = -1;
- const char *library_file;
- void *library_handle;
- library_file = (*env)->GetStringUTFChars(env, library, NULL);
- library_handle = dlopen(library_file, RTLD_GLOBAL);
- if (library_handle == NULL) {
- /* When deploying android app bundle format uncompressed native libs may not extract from apk to filesystem.
- In this case we should use lib name without path. https://bugzilla.libsdl.org/show_bug.cgi?id=4739 */
- const char *library_name = SDL_strrchr(library_file, '/');
- if (library_name && *library_name) {
- library_name += 1;
- library_handle = dlopen(library_name, RTLD_GLOBAL);
- }
- }
- if (library_handle) {
- const char *function_name;
- SDL_main_func SDL_main;
- function_name = (*env)->GetStringUTFChars(env, function, NULL);
- SDL_main = (SDL_main_func)dlsym(library_handle, function_name);
- if (SDL_main) {
- // Use the name "app_process" for argv[0] so PHYSFS_platformCalcBaseDir() works.
- // https://github.com/love2d/love-android/issues/24
- // (note that PhysicsFS hasn't used argv on Android in a long time, but we'll keep this for compat at least for SDL3's lifetime. --ryan.)
- const char *argv0 = "app_process";
- const int len = (*env)->GetArrayLength(env, array); // argv elements, not counting argv[0].
- size_t total_alloc_len = (SDL_strlen(argv0) + 1) + ((len + 2) * sizeof (char *)); // len+2 to allocate an array that also holds argv0 and a NULL terminator.
- for (int i = 0; i < len; ++i) {
- total_alloc_len++; // null terminator.
- jstring string = (*env)->GetObjectArrayElement(env, array, i);
- if (string) {
- const char *utf = (*env)->GetStringUTFChars(env, string, 0);
- if (utf) {
- total_alloc_len += SDL_strlen(utf) + 1;
- (*env)->ReleaseStringUTFChars(env, string, utf);
- }
- (*env)->DeleteLocalRef(env, string);
- }
- }
- void *args = malloc(total_alloc_len); // This should NOT be SDL_malloc()
- if (!args) { // uhoh.
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "nativeRunMain(): Out of memory parsing command line!");
- } else {
- size_t remain = total_alloc_len - (sizeof (char *) * (len + 2));
- int argc = 0;
- char **argv = (char **) args;
- char *ptr = (char *) &argv[len + 2];
- size_t cpy = SDL_strlcpy(ptr, argv0, remain) + 1;
- argv[argc++] = ptr;
- SDL_assert(cpy <= remain); remain -= cpy; ptr += cpy;
- for (int i = 0; i < len; ++i) {
- jstring string = (*env)->GetObjectArrayElement(env, array, i);
- const char *utf = string ? (*env)->GetStringUTFChars(env, string, 0) : NULL;
- cpy = SDL_strlcpy(ptr, utf ? utf : "", remain) + 1;
- if (cpy < remain) {
- argv[argc++] = ptr;
- remain -= cpy;
- ptr += cpy;
- }
- if (utf) {
- (*env)->ReleaseStringUTFChars(env, string, utf);
- }
- if (string) {
- (*env)->DeleteLocalRef(env, string);
- }
- }
- argv[argc] = NULL;
- // Run the application.
- status = SDL_RunApp(argc, argv, SDL_main, NULL);
- // Release the arguments.
- free(args); // This should NOT be SDL_free()
- }
- } else {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "nativeRunMain(): Couldn't find function %s in library %s", function_name, library_file);
- }
- (*env)->ReleaseStringUTFChars(env, function, function_name);
- dlclose(library_handle);
- } else {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "nativeRunMain(): Couldn't load library %s", library_file);
- }
- (*env)->ReleaseStringUTFChars(env, library, library_file);
- // Do not issue an exit or the whole application will terminate instead of just the SDL thread
- // exit(status);
- return status;
- }
- static int FindLifecycleEvent(SDL_AndroidLifecycleEvent event)
- {
- for (int index = 0; index < Android_NumLifecycleEvents; ++index) {
- if (Android_LifecycleEvents[index] == event) {
- return index;
- }
- }
- return -1;
- }
- static void RemoveLifecycleEvent(int index)
- {
- if (index < Android_NumLifecycleEvents - 1) {
- SDL_memmove(&Android_LifecycleEvents[index], &Android_LifecycleEvents[index+1], (Android_NumLifecycleEvents - index - 1) * sizeof(Android_LifecycleEvents[index]));
- }
- --Android_NumLifecycleEvents;
- }
- void Android_SendLifecycleEvent(SDL_AndroidLifecycleEvent event)
- {
- SDL_LockMutex(Android_LifecycleMutex);
- {
- int index;
- bool add_event = true;
- switch (event) {
- case SDL_ANDROID_LIFECYCLE_WAKE:
- // We don't need more than one wake queued
- index = FindLifecycleEvent(SDL_ANDROID_LIFECYCLE_WAKE);
- if (index >= 0) {
- add_event = false;
- }
- break;
- case SDL_ANDROID_LIFECYCLE_PAUSE:
- // If we have a resume queued, just stay in the paused state
- index = FindLifecycleEvent(SDL_ANDROID_LIFECYCLE_RESUME);
- if (index >= 0) {
- RemoveLifecycleEvent(index);
- add_event = false;
- }
- break;
- case SDL_ANDROID_LIFECYCLE_RESUME:
- // If we have a pause queued, just stay in the resumed state
- index = FindLifecycleEvent(SDL_ANDROID_LIFECYCLE_PAUSE);
- if (index >= 0) {
- RemoveLifecycleEvent(index);
- add_event = false;
- }
- break;
- case SDL_ANDROID_LIFECYCLE_LOWMEMORY:
- // We don't need more than one low memory event queued
- index = FindLifecycleEvent(SDL_ANDROID_LIFECYCLE_LOWMEMORY);
- if (index >= 0) {
- add_event = false;
- }
- break;
- case SDL_ANDROID_LIFECYCLE_DESTROY:
- // Remove all other events, we're done!
- while (Android_NumLifecycleEvents > 0) {
- RemoveLifecycleEvent(0);
- }
- break;
- default:
- SDL_assert(!"Sending unexpected lifecycle event");
- add_event = false;
- break;
- }
- if (add_event) {
- SDL_assert(Android_NumLifecycleEvents < SDL_arraysize(Android_LifecycleEvents));
- Android_LifecycleEvents[Android_NumLifecycleEvents++] = event;
- SDL_SignalSemaphore(Android_LifecycleEventSem);
- }
- }
- SDL_UnlockMutex(Android_LifecycleMutex);
- }
- bool Android_WaitLifecycleEvent(SDL_AndroidLifecycleEvent *event, Sint64 timeoutNS)
- {
- bool got_event = false;
- while (!got_event && SDL_WaitSemaphoreTimeoutNS(Android_LifecycleEventSem, timeoutNS)) {
- SDL_LockMutex(Android_LifecycleMutex);
- {
- if (Android_NumLifecycleEvents > 0) {
- *event = Android_LifecycleEvents[0];
- RemoveLifecycleEvent(0);
- got_event = true;
- }
- }
- SDL_UnlockMutex(Android_LifecycleMutex);
- }
- return got_event;
- }
- void Android_LockActivityMutex(void)
- {
- SDL_LockMutex(Android_ActivityMutex);
- }
- void Android_UnlockActivityMutex(void)
- {
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Drop file
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeDropFile)(
- JNIEnv *env, jclass jcls,
- jstring filename)
- {
- const char *path = (*env)->GetStringUTFChars(env, filename, NULL);
- SDL_SendDropFile(NULL, NULL, path);
- (*env)->ReleaseStringUTFChars(env, filename, path);
- SDL_SendDropComplete(NULL);
- }
- // Set screen resolution
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetScreenResolution)(
- JNIEnv *env, jclass jcls,
- jint surfaceWidth, jint surfaceHeight,
- jint deviceWidth, jint deviceHeight, jfloat density, jfloat rate)
- {
- SDL_LockMutex(Android_ActivityMutex);
- Android_SetScreenResolution(surfaceWidth, surfaceHeight, deviceWidth, deviceHeight, density, rate);
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Resize
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeResize)(
- JNIEnv *env, jclass jcls)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- Android_SendResize(Android_Window);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetNaturalOrientation)(
- JNIEnv *env, jclass jcls,
- jint orientation)
- {
- displayNaturalOrientation = (SDL_DisplayOrientation)orientation;
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeRotationChanged)(
- JNIEnv *env, jclass jcls,
- jint rotation)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (displayNaturalOrientation == SDL_ORIENTATION_LANDSCAPE) {
- rotation += 90;
- }
- switch (rotation % 360) {
- case 0:
- displayCurrentOrientation = SDL_ORIENTATION_PORTRAIT;
- break;
- case 90:
- displayCurrentOrientation = SDL_ORIENTATION_LANDSCAPE;
- break;
- case 180:
- displayCurrentOrientation = SDL_ORIENTATION_PORTRAIT_FLIPPED;
- break;
- case 270:
- displayCurrentOrientation = SDL_ORIENTATION_LANDSCAPE_FLIPPED;
- break;
- default:
- displayCurrentOrientation = SDL_ORIENTATION_UNKNOWN;
- break;
- }
- Android_SetOrientation(displayCurrentOrientation);
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeInsetsChanged)(
- JNIEnv *env, jclass jcls,
- jint left, jint right, jint top, jint bottom)
- {
- SDL_LockMutex(Android_ActivityMutex);
- Android_SetWindowSafeAreaInsets(left, right, top, bottom);
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeAddTouch)(
- JNIEnv *env, jclass cls,
- jint touchId, jstring name)
- {
- const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
- SDL_AddTouch(Android_ConvertJavaTouchID(touchId),
- SDL_TOUCH_DEVICE_DIRECT, utfname);
- (*env)->ReleaseStringUTFChars(env, name, utfname);
- }
- JNIEXPORT void JNICALL
- SDL_JAVA_AUDIO_INTERFACE(nativeAddAudioDevice)(JNIEnv *env, jclass jcls, jboolean recording,
- jstring name, jint device_id)
- {
- #if ALLOW_MULTIPLE_ANDROID_AUDIO_DEVICES
- if (SDL_GetCurrentAudioDriver() != NULL) {
- void *handle = (void *)((size_t)device_id);
- if (!SDL_FindPhysicalAudioDeviceByHandle(handle)) {
- const char *utf8name = (*env)->GetStringUTFChars(env, name, NULL);
- SDL_AddAudioDevice(recording, SDL_strdup(utf8name), NULL, handle);
- (*env)->ReleaseStringUTFChars(env, name, utf8name);
- }
- }
- #endif
- }
- JNIEXPORT void JNICALL
- SDL_JAVA_AUDIO_INTERFACE(nativeRemoveAudioDevice)(JNIEnv *env, jclass jcls, jboolean recording,
- jint device_id)
- {
- #if ALLOW_MULTIPLE_ANDROID_AUDIO_DEVICES
- if (SDL_GetCurrentAudioDriver() != NULL) {
- SDL_Log("Removing device with handle %d, recording %d", device_id, recording);
- SDL_AudioDeviceDisconnected(SDL_FindPhysicalAudioDeviceByHandle((void *)((size_t)device_id)));
- }
- #endif
- }
- // Paddown
- JNIEXPORT jboolean JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadDown)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint keycode)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- return Android_OnPadDown(device_id, keycode);
- #else
- return false;
- #endif // SDL_JOYSTICK_ANDROID
- }
- // Padup
- JNIEXPORT jboolean JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativePadUp)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint keycode)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- return Android_OnPadUp(device_id, keycode);
- #else
- return false;
- #endif // SDL_JOYSTICK_ANDROID
- }
- // Joy
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoy)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint axis, jfloat value)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- Android_OnJoy(device_id, axis, value);
- #endif // SDL_JOYSTICK_ANDROID
- }
- // POV Hat
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeHat)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint hat_id, jint x, jint y)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- Android_OnHat(device_id, hat_id, x, y);
- #endif // SDL_JOYSTICK_ANDROID
- }
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(onNativeJoySensor)(
- JNIEnv *env, jclass jcls,
- jint device_id, jint sensor_type, jlong sensor_timestamp, jfloat x, jfloat y, jfloat z)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- // In Java there's no Uint64 type, so pass Sint64 as if it was Uint64.
- Android_OnJoySensor(device_id, sensor_type, sensor_timestamp, x, y, z);
- #endif // SDL_JOYSTICK_ANDROID
- }
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddJoystick)(
- JNIEnv *env, jclass jcls,
- jint device_id, jstring device_name, jstring device_desc,
- jint vendor_id, jint product_id,
- jint button_mask, jint naxes, jint axis_mask, jint nhats, jboolean can_rumble, jboolean has_rgb_led,
- jboolean has_accelerometer, jboolean has_gyroscope)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- const char *name = (*env)->GetStringUTFChars(env, device_name, NULL);
- const char *desc = (*env)->GetStringUTFChars(env, device_desc, NULL);
- Android_AddJoystick(device_id, name, desc, vendor_id, product_id, button_mask, naxes, axis_mask, nhats,
- can_rumble, has_rgb_led, has_accelerometer, has_gyroscope);
- (*env)->ReleaseStringUTFChars(env, device_name, name);
- (*env)->ReleaseStringUTFChars(env, device_desc, desc);
- #endif // SDL_JOYSTICK_ANDROID
- }
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveJoystick)(
- JNIEnv *env, jclass jcls,
- jint device_id)
- {
- #ifdef SDL_JOYSTICK_ANDROID
- Android_RemoveJoystick(device_id);
- #endif // SDL_JOYSTICK_ANDROID
- }
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeAddHaptic)(
- JNIEnv *env, jclass jcls, jint device_id, jstring device_name)
- {
- #ifdef SDL_HAPTIC_ANDROID
- const char *name = (*env)->GetStringUTFChars(env, device_name, NULL);
- Android_AddHaptic(device_id, name);
- (*env)->ReleaseStringUTFChars(env, device_name, name);
- #endif // SDL_HAPTIC_ANDROID
- }
- JNIEXPORT void JNICALL SDL_JAVA_CONTROLLER_INTERFACE(nativeRemoveHaptic)(
- JNIEnv *env, jclass jcls, jint device_id)
- {
- #ifdef SDL_HAPTIC_ANDROID
- Android_RemoveHaptic(device_id);
- #endif
- }
- // Called from surfaceCreated()
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceCreated)(JNIEnv *env, jclass jcls)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- SDL_WindowData *data = Android_Window->internal;
- data->native_window = Android_JNI_GetNativeWindow();
- SDL_SetPointerProperty(SDL_GetWindowProperties(Android_Window), SDL_PROP_WINDOW_ANDROID_WINDOW_POINTER, data->native_window);
- if (data->native_window == NULL) {
- SDL_SetError("Could not fetch native window from UI thread");
- }
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Called from surfaceChanged()
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceChanged)(JNIEnv *env, jclass jcls)
- {
- SDL_LockMutex(Android_ActivityMutex);
- #ifdef SDL_VIDEO_OPENGL_EGL
- if (Android_Window && (Android_Window->flags & SDL_WINDOW_OPENGL)) {
- SDL_VideoDevice *_this = SDL_GetVideoDevice();
- SDL_WindowData *data = Android_Window->internal;
- // If the surface has been previously destroyed by onNativeSurfaceDestroyed, recreate it here
- if (data->egl_surface == EGL_NO_SURFACE) {
- data->egl_surface = SDL_EGL_CreateSurface(_this, Android_Window, (NativeWindowType)data->native_window);
- SDL_SetPointerProperty(SDL_GetWindowProperties(Android_Window), SDL_PROP_WINDOW_ANDROID_SURFACE_POINTER, data->egl_surface);
- }
- // GL Context handling is done in the event loop because this function is run from the Java thread
- }
- #endif
- if (Android_Window) {
- Android_RestoreScreenKeyboard(SDL_GetVideoDevice(), Android_Window);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Called from surfaceDestroyed()
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeSurfaceDestroyed)(JNIEnv *env, jclass jcls)
- {
- int nb_attempt = 50;
- retry:
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- SDL_WindowData *data = Android_Window->internal;
- // Wait for Main thread being paused and context un-activated to release 'egl_surface'
- if ((Android_Window->flags & SDL_WINDOW_OPENGL) && !data->backup_done) {
- nb_attempt -= 1;
- if (nb_attempt == 0) {
- SDL_SetError("Try to release egl_surface with context probably still active");
- } else {
- SDL_UnlockMutex(Android_ActivityMutex);
- SDL_Delay(10);
- goto retry;
- }
- }
- #ifdef SDL_VIDEO_OPENGL_EGL
- if (data->egl_surface != EGL_NO_SURFACE) {
- SDL_EGL_DestroySurface(SDL_GetVideoDevice(), data->egl_surface);
- data->egl_surface = EGL_NO_SURFACE;
- }
- #endif
- if (data->native_window) {
- ANativeWindow_release(data->native_window);
- data->native_window = NULL;
- }
- // GL Context handling is done in the event loop because this function is run from the Java thread
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeScreenKeyboardShown)(JNIEnv *env, jclass jcls)
- {
- SDL_SendScreenKeyboardShown();
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeScreenKeyboardHidden)(JNIEnv *env, jclass jcls)
- {
- SDL_SendScreenKeyboardHidden();
- }
- // Keydown
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyDown)(
- JNIEnv *env, jclass jcls,
- jint keycode)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- Android_OnKeyDown(keycode);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Keyup
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyUp)(
- JNIEnv *env, jclass jcls,
- jint keycode)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- Android_OnKeyUp(keycode);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Virtual keyboard return key might stop text input
- JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(onNativeSoftReturnKey)(
- JNIEnv *env, jclass jcls)
- {
- if (SDL_GetHintBoolean(SDL_HINT_RETURN_KEY_HIDES_IME, false)) {
- SDL_StopTextInput(Android_Window);
- return JNI_TRUE;
- }
- return JNI_FALSE;
- }
- // Keyboard Focus Lost
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeKeyboardFocusLost)(
- JNIEnv *env, jclass jcls)
- {
- // Calling SDL_StopTextInput will take care of hiding the keyboard and cleaning up the DummyText widget
- SDL_StopTextInput(Android_Window);
- }
- // Touch
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeTouch)(
- JNIEnv *env, jclass jcls,
- jint touch_device_id_in, jint pointer_finger_id_in,
- jint action, jfloat x, jfloat y, jfloat p)
- {
- SDL_LockMutex(Android_ActivityMutex);
- Android_OnTouch(Android_Window, touch_device_id_in, pointer_finger_id_in, action, x, y, p);
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Pinch
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePinchStart)(
- JNIEnv *env, jclass jcls)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- SDL_SendPinch(SDL_EVENT_PINCH_BEGIN, 0, Android_Window, 0);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePinchUpdate)(
- JNIEnv *env, jclass jcls, jfloat scale)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- SDL_SendPinch(SDL_EVENT_PINCH_UPDATE, 0, Android_Window, scale);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePinchEnd)(
- JNIEnv *env, jclass jcls)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- SDL_SendPinch(SDL_EVENT_PINCH_END, 0, Android_Window, 0);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Mouse
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeMouse)(
- JNIEnv *env, jclass jcls,
- jint button, jint action, jfloat x, jfloat y, jboolean relative)
- {
- SDL_LockMutex(Android_ActivityMutex);
- Android_OnMouse(Android_Window, button, action, x, y, relative);
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Pen
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativePen)(
- JNIEnv *env, jclass jcls,
- jint pen_id_in, jint device_type, jint button, jint action, jfloat x, jfloat y, jfloat p)
- {
- SDL_LockMutex(Android_ActivityMutex);
- Android_OnPen(Android_Window, pen_id_in, device_type, button, action, x, y, p);
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- // Accelerometer
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeAccel)(
- JNIEnv *env, jclass jcls,
- jfloat x, jfloat y, jfloat z)
- {
- fLastAccelerometer[0] = x;
- fLastAccelerometer[1] = y;
- fLastAccelerometer[2] = z;
- bHasNewData = true;
- }
- // Clipboard
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeClipboardChanged)(
- JNIEnv *env, jclass jcls)
- {
- // TODO: compute new mime types
- SDL_SendClipboardUpdate(false, NULL, 0);
- }
- // Low memory
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeLowMemory)(
- JNIEnv *env, jclass cls)
- {
- Android_SendLifecycleEvent(SDL_ANDROID_LIFECYCLE_LOWMEMORY);
- }
- /* Locale
- * requires android:configChanges="layoutDirection|locale" in AndroidManifest.xml */
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeLocaleChanged)(
- JNIEnv *env, jclass cls)
- {
- SDL_SendAppEvent(SDL_EVENT_LOCALE_CHANGED);
- }
- // Dark mode
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeDarkModeChanged)(
- JNIEnv *env, jclass cls, jboolean enabled)
- {
- Android_SetDarkMode(enabled);
- }
- // Send Quit event to "SDLThread" thread
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSendQuit)(
- JNIEnv *env, jclass cls)
- {
- Android_SendLifecycleEvent(SDL_ANDROID_LIFECYCLE_DESTROY);
- }
- // Activity ends
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeQuit)(
- JNIEnv *env, jclass cls)
- {
- const char *str;
- if (Android_ActivityMutex) {
- SDL_DestroyMutex(Android_ActivityMutex);
- Android_ActivityMutex = NULL;
- }
- if (Android_LifecycleMutex) {
- SDL_DestroyMutex(Android_LifecycleMutex);
- Android_LifecycleMutex = NULL;
- }
- if (Android_LifecycleEventSem) {
- SDL_DestroySemaphore(Android_LifecycleEventSem);
- Android_LifecycleEventSem = NULL;
- }
- Android_NumLifecycleEvents = 0;
- Internal_Android_Destroy_AssetManager();
- str = SDL_GetError();
- if (str && str[0]) {
- __android_log_print(ANDROID_LOG_ERROR, "SDL", "SDLActivity thread ends (error=%s)", str);
- } else {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "SDLActivity thread ends");
- }
- }
- // Pause
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePause)(
- JNIEnv *env, jclass cls)
- {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativePause()");
- Android_SendLifecycleEvent(SDL_ANDROID_LIFECYCLE_PAUSE);
- }
- // Resume
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeResume)(
- JNIEnv *env, jclass cls)
- {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeResume()");
- Android_SendLifecycleEvent(SDL_ANDROID_LIFECYCLE_RESUME);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeFocusChanged)(
- JNIEnv *env, jclass cls, jboolean hasFocus)
- {
- SDL_LockMutex(Android_ActivityMutex);
- if (Android_Window) {
- __android_log_print(ANDROID_LOG_VERBOSE, "SDL", "nativeFocusChanged()");
- SDL_SendWindowEvent(Android_Window, (hasFocus ? SDL_EVENT_WINDOW_FOCUS_GAINED : SDL_EVENT_WINDOW_FOCUS_LOST), 0, 0);
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeCommitText)(
- JNIEnv *env, jclass cls,
- jstring text, jint newCursorPosition)
- {
- const char *utftext = (*env)->GetStringUTFChars(env, text, NULL);
- SDL_SendKeyboardText(utftext);
- (*env)->ReleaseStringUTFChars(env, text, utftext);
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE_INPUT_CONNECTION(nativeGenerateScancodeForUnichar)(
- JNIEnv *env, jclass cls,
- jchar chUnicode)
- {
- SDL_SendKeyboardUnicodeKey(0, chUnicode);
- }
- JNIEXPORT jstring JNICALL SDL_JAVA_INTERFACE(nativeGetHint)(
- JNIEnv *env, jclass cls,
- jstring name)
- {
- const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
- const char *hint = SDL_GetHint(utfname);
- jstring result = (*env)->NewStringUTF(env, hint);
- (*env)->ReleaseStringUTFChars(env, name, utfname);
- return result;
- }
- JNIEXPORT jboolean JNICALL SDL_JAVA_INTERFACE(nativeGetHintBoolean)(
- JNIEnv *env, jclass cls,
- jstring name, jboolean default_value)
- {
- jboolean result;
- const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
- result = SDL_GetHintBoolean(utfname, default_value);
- (*env)->ReleaseStringUTFChars(env, name, utfname);
- return result;
- }
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativeSetenv)(
- JNIEnv *env, jclass cls,
- jstring name, jstring value)
- {
- const char *utfname = (*env)->GetStringUTFChars(env, name, NULL);
- const char *utfvalue = (*env)->GetStringUTFChars(env, value, NULL);
- // This is only called at startup, to initialize the environment
- // Note that we call setenv() directly to avoid affecting SDL environments
- setenv(utfname, utfvalue, 1); // This should NOT be SDL_setenv()
- if (SDL_strcmp(utfname, SDL_HINT_ANDROID_ALLOW_RECREATE_ACTIVITY) == 0) {
- // Special handling for this hint, which needs to persist outside the normal application flow
- // Only set this the first time we run, in case it's been set by the application via SDL_SetHint()
- if (!allow_recreate_activity_set) {
- Android_SetAllowRecreateActivity(SDL_GetStringBoolean(utfvalue, false));
- }
- }
- (*env)->ReleaseStringUTFChars(env, name, utfname);
- (*env)->ReleaseStringUTFChars(env, value, utfvalue);
- }
- /*******************************************************************************
- Functions called by SDL into Java
- *******************************************************************************/
- static SDL_AtomicInt s_active;
- struct LocalReferenceHolder
- {
- JNIEnv *m_env;
- const char *m_func;
- };
- static struct LocalReferenceHolder LocalReferenceHolder_Setup(const char *func)
- {
- struct LocalReferenceHolder refholder;
- refholder.m_env = NULL;
- refholder.m_func = func;
- #ifdef DEBUG_JNI
- SDL_Log("Entering function %s", func);
- #endif
- return refholder;
- }
- static bool LocalReferenceHolder_Init(struct LocalReferenceHolder *refholder, JNIEnv *env)
- {
- const int capacity = 16;
- if ((*env)->PushLocalFrame(env, capacity) < 0) {
- SDL_SetError("Failed to allocate enough JVM local references");
- return false;
- }
- SDL_AtomicIncRef(&s_active);
- refholder->m_env = env;
- return true;
- }
- static void LocalReferenceHolder_Cleanup(struct LocalReferenceHolder *refholder)
- {
- #ifdef DEBUG_JNI
- SDL_Log("Leaving function %s", refholder->m_func);
- #endif
- if (refholder->m_env) {
- JNIEnv *env = refholder->m_env;
- (*env)->PopLocalFrame(env, NULL);
- SDL_AtomicDecRef(&s_active);
- }
- }
- ANativeWindow *Android_JNI_GetNativeWindow(void)
- {
- ANativeWindow *anw = NULL;
- jobject s;
- JNIEnv *env = Android_JNI_GetEnv();
- s = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetNativeSurface);
- if (s) {
- anw = ANativeWindow_fromSurface(env, s);
- (*env)->DeleteLocalRef(env, s);
- }
- return anw;
- }
- void Android_JNI_SetActivityTitle(const char *title)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- jstring jtitle = (*env)->NewStringUTF(env, title);
- (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetActivityTitle, jtitle);
- (*env)->DeleteLocalRef(env, jtitle);
- }
- void Android_JNI_SetWindowStyle(bool fullscreen)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mActivityClass, midSetWindowStyle, fullscreen ? 1 : 0);
- }
- void Android_JNI_SetOrientation(int w, int h, int resizable, const char *hint)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- jstring jhint = (*env)->NewStringUTF(env, (hint ? hint : ""));
- (*env)->CallStaticVoidMethod(env, mActivityClass, midSetOrientation, w, h, (resizable ? 1 : 0), jhint);
- (*env)->DeleteLocalRef(env, jhint);
- }
- SDL_DisplayOrientation Android_JNI_GetDisplayNaturalOrientation(void)
- {
- return displayNaturalOrientation;
- }
- SDL_DisplayOrientation Android_JNI_GetDisplayCurrentOrientation(void)
- {
- return displayCurrentOrientation;
- }
- void Android_JNI_MinimizeWindow(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mActivityClass, midMinimizeWindow);
- }
- bool Android_JNI_ShouldMinimizeOnFocusLoss(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midShouldMinimizeOnFocusLoss);
- }
- bool Android_JNI_GetAccelerometerValues(float values[3])
- {
- bool result = false;
- if (bHasNewData) {
- int i;
- for (i = 0; i < 3; ++i) {
- values[i] = fLastAccelerometer[i];
- }
- bHasNewData = false;
- result = true;
- }
- return result;
- }
- /*
- * Audio support
- */
- void Android_StartAudioHotplug(SDL_AudioDevice **default_playback, SDL_AudioDevice **default_recording)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- // this will fire the callback for each existing device right away (which will eventually SDL_AddAudioDevice), and again later when things change.
- (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midRegisterAudioDeviceCallback);
- *default_playback = *default_recording = NULL; // !!! FIXME: how do you decide the default device id?
- }
- void Android_StopAudioHotplug(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midUnregisterAudioDeviceCallback);
- }
- static void Android_JNI_AudioSetThreadPriority(int recording, int device_id)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mAudioManagerClass, midAudioSetThreadPriority, recording, device_id);
- }
- void Android_AudioThreadInit(SDL_AudioDevice *device)
- {
- Android_JNI_AudioSetThreadPriority((int) device->recording, (int)device->instance_id);
- }
- // Test for an exception and call SDL_SetError with its detail if one occurs
- // If the parameter silent is truthy then SDL_SetError() will not be called.
- static bool Android_JNI_ExceptionOccurred(bool silent)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- jthrowable exception;
- // Detect mismatch LocalReferenceHolder_Init/Cleanup
- SDL_assert(SDL_GetAtomicInt(&s_active) > 0);
- exception = (*env)->ExceptionOccurred(env);
- if (exception != NULL) {
- jmethodID mid;
- // Until this happens most JNI operations have undefined behaviour
- (*env)->ExceptionClear(env);
- if (!silent) {
- jclass exceptionClass = (*env)->GetObjectClass(env, exception);
- jclass classClass = (*env)->FindClass(env, "java/lang/Class");
- jstring exceptionName;
- const char *exceptionNameUTF8;
- jstring exceptionMessage;
- mid = (*env)->GetMethodID(env, classClass, "getName", "()Ljava/lang/String;");
- exceptionName = (jstring)(*env)->CallObjectMethod(env, exceptionClass, mid);
- exceptionNameUTF8 = (*env)->GetStringUTFChars(env, exceptionName, 0);
- (*env)->DeleteLocalRef(env, classClass);
- mid = (*env)->GetMethodID(env, exceptionClass, "getMessage", "()Ljava/lang/String;");
- exceptionMessage = (jstring)(*env)->CallObjectMethod(env, exception, mid);
- (*env)->DeleteLocalRef(env, exceptionClass);
- if (exceptionMessage != NULL) {
- const char *exceptionMessageUTF8 = (*env)->GetStringUTFChars(env, exceptionMessage, 0);
- SDL_SetError("%s: %s", exceptionNameUTF8, exceptionMessageUTF8);
- (*env)->ReleaseStringUTFChars(env, exceptionMessage, exceptionMessageUTF8);
- (*env)->DeleteLocalRef(env, exceptionMessage);
- } else {
- SDL_SetError("%s", exceptionNameUTF8);
- }
- (*env)->ReleaseStringUTFChars(env, exceptionName, exceptionNameUTF8);
- (*env)->DeleteLocalRef(env, exceptionName);
- }
- return true;
- }
- return false;
- }
- // APK file tree discovery...
- // APK files are just .zip files, so try to find parse out the file tree from it. We'll still
- // use Android's system APIs to actually access data, but those APIs aren't reliable for
- // enumerating the tree for various reasons.
- // we only care about the directories; Android's AAssetManager can enumerate all files in any
- // known directory, but it won't enumerate subdirectories, so we track that by ourselves.
- typedef struct APKNode
- {
- char *name;
- SDL_PathInfo info;
- struct APKNode *children;
- struct APKNode *next_sibling;
- } APKNode;
- static APKNode *APKRootNode = NULL;
- static void FreeAPKNode(APKNode *node)
- {
- if (node) {
- FreeAPKNode(node->next_sibling);
- FreeAPKNode(node->children);
- SDL_free(node->name);
- SDL_free(node);
- }
- }
- static APKNode *FindAPKChildNode(APKNode *parent, const char *child)
- {
- for (APKNode *node = parent->children; node != NULL; node = node->next_sibling) {
- if (SDL_strcmp(child, node->name) == 0) {
- return node;
- }
- }
- return NULL;
- }
- static const APKNode *FindAPKNode(const char *constpath)
- {
- APKNode *parent = APKRootNode;
- if (!parent) {
- return NULL;
- }
- const size_t pathlen = SDL_strlen(constpath);
- bool isstack = false;
- char *alloc_path = SDL_small_alloc(char, pathlen + 1, &isstack);
- if (!alloc_path) {
- return NULL;
- }
- char *path = alloc_path;
- SDL_strlcpy(path, constpath, pathlen + 1);
- while (parent) {
- while (*path == '/') {
- path++; // just in case there are absolute paths or double-slashes, drop them.
- }
- if (*path == '\0') { // ended with a '/'? We're done.
- break;
- }
- char *ptr = SDL_strchr(path, '/');
- if (ptr) {
- *ptr = '\0'; // terminate on the end of this subdir's name.
- }
- APKNode *node = FindAPKChildNode(parent, path);
- if (!node) {
- SDL_SetError("No such file or directory");
- parent = NULL;
- } else if ((node->info.type == SDL_PATHTYPE_FILE) && ptr) { // file where we want a directory?
- SDL_SetError("%s is not a directory", alloc_path);
- parent = NULL;
- } else {
- parent = node;
- if (!ptr) {
- break;
- }
- *ptr = '/';
- path = ptr + 1;
- }
- }
- SDL_small_free(alloc_path, isstack);
- return parent;
- }
- static APKNode *AddAPKChildNode(APKNode *parent, const char *child)
- {
- APKNode *node = FindAPKChildNode(parent, child);
- if (!node) { // don't have this one yet, make a new node.
- node = (APKNode *) SDL_calloc(1, sizeof (*node));
- if (!node) {
- return NULL; // uhoh.
- }
- node->name = SDL_strdup(child);
- if (!node->name) {
- SDL_free(node);
- return NULL; // uhoh.
- }
- SDL_copyp(&node->info, &parent->info); // you probably need to update this afterwards.
- node->next_sibling = parent->children;
- parent->children = node;
- }
- return node;
- }
- static APKNode *AddAPKDirs(char *path, APKNode *parent)
- {
- // zip files specify explicit directories by just having a path that ends with a dir separator,
- // which works nicely for our needs here; if the last segment of the path doesn't end with a
- // '/', it's a file and we can drop it, or we filled in the final subdirectory and the '/' at
- // the end will put us at an empty string to be dropped.
- //
- // directories do not need to be explicitly specified if something uses some deeper path, they
- // may need to be inferred from those references, so we build out the tree by looking at all
- // files and filling in nodes they mention.
- SDL_assert(parent->info.type == SDL_PATHTYPE_DIRECTORY);
- SDL_assert(parent->info.size == 0);
- while (true) { // while still subdirectories to handle...
- while (*path == '/') {
- path++; // just in case there are absolute paths or double-slashes, drop them.
- }
- char *ptr = SDL_strchr(path, '/');
- if (!ptr) {
- break; // last thing is either an empty string (we ended with a '/'), or an actual file's name, so drop it.
- }
- *ptr = '\0'; // terminate on the end of this subdir's name.
- APKNode *node = AddAPKChildNode(parent, path);
- *ptr = '/';
- if (!node) {
- return NULL; // uhoh.
- }
- parent = node;
- path = ptr + 1; // point to start of next section.
- }
- return parent;
- }
- static SDL_Time ZipDosTimeToSDLTime(Uint32 dostime)
- {
- Uint32 dosdate;
- struct tm unixtime;
- SDL_zero(unixtime);
- dosdate = (Uint32) ((dostime >> 16) & 0xFFFF);
- dostime &= 0xFFFF;
- /* dissect date */
- unixtime.tm_year = ((dosdate >> 9) & 0x7F) + 80;
- unixtime.tm_mon = ((dosdate >> 5) & 0x0F) - 1;
- unixtime.tm_mday = ((dosdate ) & 0x1F);
- /* dissect time */
- unixtime.tm_hour = ((dostime >> 11) & 0x1F);
- unixtime.tm_min = ((dostime >> 5) & 0x3F);
- unixtime.tm_sec = ((dostime << 1) & 0x3E);
- /* let mktime calculate daylight savings time. */
- unixtime.tm_isdst = -1;
- return ((SDL_Time) mktime(&unixtime));
- }
- #define ZIP_CENTRAL_DIR_SIG 0x02014b50
- #define ZIP_END_OF_CENTRAL_DIR_SIG 0x06054b50
- #define ZIP64_END_OF_CENTRAL_DIR_SIG 0x06064b50
- #define ZIP64_END_OF_CENTRAL_DIRECTORY_LOCATOR_SIG 0x07064b50
- #define ZIP64_EXTENDED_INFO_EXTRA_FIELD_SIG 0x0001
- static bool ProcessZip(SDL_IOStream *io, APKNode *root)
- {
- // There is a record at the end of a .zip file we can use to find the central directory;
- // unfortunately it's before the variable-length comment field, so we might need to read
- // from the end of the file until we see a magic signature. For now, we assume APKs don't
- // have a comment, so we don't have to search backwards for the signature and it's always
- // 22 bytes from EOF.
- const Sint64 eocd = SDL_GetIOSize(io) - 22;
- if (eocd < 0) {
- SDL_Log("ANDROID: Couldn't find End Of Central Directory in APK (%s). Filesystem enumeration will fail.", SDL_GetError());
- return false;
- }
- bool zip64 = false;
- Sint64 centraldir = -1;
- Uint64 num_entries = 0;
- Uint16 val16 = 0;
- Uint32 val32 = 0;
- Uint64 val64 = 0;
- // First, check if this is actually zip64 format instead. The zip64 magic is 20 bytes back.
- if (eocd < 20) { // presumably we always _are_ > 20, but let's be defensive here.
- goto corrupterr;
- } else if (SDL_SeekIO(io, eocd - 20, SDL_IO_SEEK_SET) < 0) {
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) {
- goto ioerr;
- } else if (val32 == ZIP64_END_OF_CENTRAL_DIRECTORY_LOCATOR_SIG) { // this is a zip64 archive?
- if (!SDL_ReadU32LE(io, &val32)) { // disk number with start of central directory.
- goto ioerr;
- } else if (val32 != 0) {
- goto corrupterr;
- } else if (!SDL_ReadU64LE(io, &val64)) { // file offset of zip64 end-of-central-dir record
- goto ioerr;
- // note that this gets significantly more complex if there is data prepended to the .zip file
- // (like a self-extracting .exe, etc), but until that happens, we're keeping this as simple
- // as possible and assuming the file offset in val64 is correct.
- } else if (SDL_SeekIO(io, (Sint64) val64, SDL_IO_SEEK_SET) < 0) {
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // zip64 end-of-central-dir signature.
- goto ioerr;
- } else if (val32 != ZIP64_END_OF_CENTRAL_DIR_SIG) {
- goto corrupterr;
- } else if (SDL_SeekIO(io, 28, SDL_IO_SEEK_CUR) < 0) { // we don't care about several of the fields, skip over them.
- goto ioerr;
- } else if (!SDL_ReadU64LE(io, &num_entries)) { // total entries in the central dir.
- goto ioerr;
- } else if (!SDL_ReadU64LE(io, &val64)) { // size of the central dir.
- goto ioerr;
- } else if (!SDL_ReadU64LE(io, &val64)) { // offset of the central dir.
- goto ioerr;
- }
- zip64 = true;
- centraldir = (Sint64) val64;
- } else if (SDL_SeekIO(io, eocd + 4 + 6, SDL_IO_SEEK_SET) < 0) { // skip back to where we were, plus skip some fields we don't care about.
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &val16)) {
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // size of the central dir.
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // offset of the central dir.
- goto ioerr;
- } else {
- num_entries = (Uint64) val16;
- centraldir = (Sint64) val32;
- }
- // okay, we know where the central dir is now, go there and start reading entries.
- SDL_assert(centraldir > 0); // negative means we failed, zero is impossible since there should be something else there.
- for (Uint64 i = 0; i < num_entries; i++) {
- Uint16 fnamelen = 0;
- Uint16 extralen = 0;
- Uint16 commentlen = 0;
- Uint32 dosmodtime = 0;
- Uint32 uncompressed32 = 0;
- Uint64 uncompressed64 = 0;
- // we don't care about most of this information, just parse through it to get what we need.
- if (SDL_SeekIO(io, centraldir, SDL_IO_SEEK_SET) < 0) {
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // central dir item signature.
- goto ioerr;
- } else if (val32 != ZIP_CENTRAL_DIR_SIG) {
- goto corrupterr;
- } else if (!SDL_ReadU16LE(io, &val16)) { // version made by
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &val16)) { // version needed
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &val16)) { // general bits
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &val16)) { // compression method
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &dosmodtime)) { // last mod date/time
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // CRC-32
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // compressed size
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &uncompressed32)) { // uncompressed size
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &fnamelen)) { // filename length
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &extralen)) { // extra length
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &commentlen)) { // comment length
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &val16)) { // disk number start
- goto ioerr;
- } else if (!SDL_ReadU16LE(io, &val16)) { // internal file attributes
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // external file attributes
- goto ioerr;
- } else if (!SDL_ReadU32LE(io, &val32)) { // relative offset of local header
- goto ioerr;
- }
- char fnamebuf[0xFFFF+1]; // just eat 64k of stack like a boss until someone complains.
- if (SDL_ReadIO(io, fnamebuf, (size_t) fnamelen) != ((size_t) fnamelen)) {
- goto ioerr;
- }
- // technically zip files might have '\\' dir separators, but these were mostly old DOS files and not Android APKs, I think. Revisit if necessary.
- fnamebuf[fnamelen] = '\0'; // make sure the string is null-terminated.
- //SDL_Log("ANDROID: Saw ZIP entry '%s'", fnamebuf);
- uncompressed64 = (Uint64) uncompressed32;
- if (zip64 && (uncompressed32 == 0xFFFFFFFF)) { // file is larger than 4gig, find the zip64 extended info field in the extra section.
- bool found = false;
- Uint16 remaining = extralen;
- while (remaining > 4) { // Two 16-bit values at a minimum, tag and len.
- Uint16 tag, len;
- if (!SDL_ReadU16LE(io, &tag) || !SDL_ReadU16LE(io, &len)) {
- goto ioerr;
- } else if (remaining < (len + 4)) {
- goto corrupterr;
- } else if (tag != ZIP64_EXTENDED_INFO_EXTRA_FIELD_SIG) { // not the field we need, skip over it.
- if (SDL_SeekIO(io, (Sint64) len, SDL_IO_SEEK_CUR) < 0) {
- goto ioerr;
- }
- } else if (len < 8) {
- goto corrupterr;
- } else if ((uncompressed32 == 0xFFFFFFFF) && !SDL_ReadU64LE(io, &uncompressed64)) {
- goto ioerr;
- // there are other values in here, but we don't care about them and we're done, so don't try to skip over them.
- } else {
- found = true;
- break; // got what we need, drop out.
- }
- remaining -= len + 4;
- }
- if (!found) {
- goto corrupterr;
- }
- }
- char *ptr = fnamebuf;
- while (*ptr == '/') { // drop absolute paths.
- ptr++;
- }
- if (SDL_strncmp(ptr, "assets/", 7) == 0) { // we only care about things under 'assets' for now. Drop everything else.
- ptr += 6; // keep the '/' so strrchr never returns NULL.
- APKNode *node = AddAPKDirs(ptr, root); // this builds out any missing subdirs, returns parent dir's node.
- if (!node) {
- goto ioerr; // (probably out of memory.)
- }
- const SDL_Time modtime = ZipDosTimeToSDLTime(dosmodtime);
- ptr = SDL_strrchr(ptr, '/');
- SDL_assert(ptr != NULL);
- if (*(++ptr) == '\0') { // explicit directory entry paths end with '/' ...`node` is the new node.
- node->info.type = SDL_PATHTYPE_DIRECTORY;
- node->info.size = 0;
- } else {
- node = AddAPKChildNode(node, ptr);
- if (!node) {
- goto ioerr; // (probably out of memory.)
- }
- node->info.type = SDL_PATHTYPE_FILE;
- node->info.size = uncompressed64;
- }
- node->info.create_time = node->info.modify_time = node->info.access_time = modtime;
- }
- centraldir += (Sint64) (46 + fnamelen + extralen + commentlen); // will seek to next file entry.
- }
- return true;
- corrupterr:
- SDL_Log("ANDROID: Unexpected or corrupt data in APK. Filesystem enumeration will fail.");
- return false;
- ioerr:
- SDL_Log("ANDROID: i/o error in APK (%s). Filesystem enumeration will fail.", SDL_GetError());
- return false;
- }
- static bool CreateAPKNodes(const char *path)
- {
- SDL_Log("ANDROID: Parsing APK file '%s' ...", path);
- SDL_IOStream *io = SDL_IOFromFile(path, "rb");
- if (!io) {
- SDL_Log("ANDROID: Can't open APK '%s' for reading (%s). Filesystem enumeration will fail.", path, SDL_GetError());
- } else {
- ProcessZip(io, APKRootNode);
- SDL_CloseIO(io);
- }
- return true;
- }
- static bool PrepareAPK(void)
- {
- // the assetmanager isn't useful for enumerating directories, so parse the APK directly for that info upfront.
- bool retval = (APKRootNode != NULL);
- if (!retval) {
- // allocate this upfront, so if there's a failure, we'll not try again and just have an empty file tree.
- APKRootNode = (APKNode *) SDL_calloc(1, sizeof (*APKRootNode));
- if (!APKRootNode) {
- return false; // oh well.
- }
- APKRootNode->info.type = SDL_PATHTYPE_DIRECTORY;
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- JNIEnv *env = Android_JNI_GetEnv();
- if (LocalReferenceHolder_Init(&refs, env)) {
- jobject context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- jmethodID mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context), "getPackageResourcePath", "()Ljava/lang/String;");
- jstring jstr = (jstring)(*env)->CallObjectMethod(env, context, mid);
- jthrowable jexception = (*env)->ExceptionOccurred(env);
- if (jexception != NULL) {
- (*env)->ExceptionClear(env); // oh well
- } else {
- const char *apkpath = (*env)->GetStringUTFChars(env, jstr, NULL);
- SDL_PathInfo apkinfo;
- SDL_assert(apkpath[0] == '/'); // So SDL_GetPathInfo goes through the `stat` path and doesn't try to dig into the APK.
- if (SDL_GetPathInfo(apkpath, &apkinfo)) { // we just want the file times here, so oh well if it fails.
- APKRootNode->info.create_time = apkinfo.create_time;
- APKRootNode->info.modify_time = apkinfo.modify_time;
- APKRootNode->info.access_time = apkinfo.access_time;
- }
- CreateAPKNodes(apkpath);
- (*env)->ReleaseStringUTFChars(env, jstr, apkpath);
- retval = true;
- }
- }
- LocalReferenceHolder_Cleanup(&refs);
- }
- return retval; // even on failure, leave an empty root node so we have zero files and don't try to load the .zip again.
- }
- static void Internal_Android_Create_AssetManager(void)
- {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- JNIEnv *env = Android_JNI_GetEnv();
- jmethodID mid;
- jobject context;
- jobject javaAssetManager;
- if (!LocalReferenceHolder_Init(&refs, env)) {
- LocalReferenceHolder_Cleanup(&refs);
- return;
- }
- // context = SDLActivity.getContext();
- context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- // javaAssetManager = context.getAssets();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
- "getAssets", "()Landroid/content/res/AssetManager;");
- javaAssetManager = (*env)->CallObjectMethod(env, context, mid);
- /**
- * Given a Dalvik AssetManager object, obtain the corresponding native AAssetManager
- * object. Note that the caller is responsible for obtaining and holding a VM reference
- * to the jobject to prevent its being garbage collected while the native object is
- * in use.
- */
- javaAssetManagerRef = (*env)->NewGlobalRef(env, javaAssetManager);
- asset_manager = AAssetManager_fromJava(env, javaAssetManagerRef);
- if (!asset_manager) {
- (*env)->DeleteGlobalRef(env, javaAssetManagerRef);
- Android_JNI_ExceptionOccurred(true);
- }
- LocalReferenceHolder_Cleanup(&refs);
- }
- static void Internal_Android_Destroy_AssetManager(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- if (asset_manager) {
- (*env)->DeleteGlobalRef(env, javaAssetManagerRef);
- asset_manager = NULL;
- }
- if (APKRootNode) {
- FreeAPKNode(APKRootNode);
- APKRootNode = NULL;
- }
- }
- static const char *GetAssetPath(const char *path)
- {
- if (path && path[0] == '.' && path[1] == '/') {
- path += 2;
- while (*path == '/') {
- ++path;
- }
- }
- return path;
- }
- bool Android_JNI_FileOpen(void **puserdata, const char *fileName, const char *mode)
- {
- SDL_assert(puserdata != NULL);
- AAsset *asset = NULL;
- *puserdata = NULL;
- if (!asset_manager) {
- Internal_Android_Create_AssetManager();
- if (!asset_manager) {
- return SDL_SetError("Couldn't create asset manager");
- }
- }
- fileName = GetAssetPath(fileName);
- asset = AAssetManager_open(asset_manager, fileName, AASSET_MODE_UNKNOWN);
- if (!asset) {
- return SDL_SetError("Couldn't open asset '%s'", fileName);
- }
- *puserdata = (void *)asset;
- return true;
- }
- size_t Android_JNI_FileRead(void *userdata, void *buffer, size_t size, SDL_IOStatus *status)
- {
- const int bytes = AAsset_read((AAsset *)userdata, buffer, size);
- if (bytes < 0) {
- SDL_SetError("AAsset_read() failed");
- *status = SDL_IO_STATUS_ERROR;
- return 0;
- } else if (bytes < size) {
- *status = SDL_IO_STATUS_EOF;
- }
- return (size_t)bytes;
- }
- size_t Android_JNI_FileWrite(void *userdata, const void *buffer, size_t size, SDL_IOStatus *status)
- {
- SDL_SetError("Cannot write to Android package filesystem");
- *status = SDL_IO_STATUS_ERROR;
- return 0;
- }
- Sint64 Android_JNI_FileSize(void *userdata)
- {
- return (Sint64) AAsset_getLength64((AAsset *)userdata);
- }
- Sint64 Android_JNI_FileSeek(void *userdata, Sint64 offset, SDL_IOWhence whence)
- {
- return (Sint64) AAsset_seek64((AAsset *)userdata, offset, (int)whence);
- }
- bool Android_JNI_FileClose(void *userdata)
- {
- AAsset_close((AAsset *)userdata);
- return true;
- }
- bool Android_JNI_EnumerateAssetDirectory(const char *path, SDL_EnumerateDirectoryCallback cb, void *userdata)
- {
- SDL_assert(path != NULL);
- if (!PrepareAPK()) {
- return false;
- }
- SDL_EnumerationResult result = SDL_ENUM_CONTINUE;
- const char *asset_path = GetAssetPath(path);
- const APKNode *apknode = FindAPKNode(asset_path);
- if (!apknode) {
- return SDL_SetError("No such directory");
- } else if (apknode->info.type != SDL_PATHTYPE_DIRECTORY) {
- return SDL_SetError("Not a directory");
- } else {
- for (const APKNode *node = apknode->children; node && (result == SDL_ENUM_CONTINUE); node = node->next_sibling) {
- result = cb(userdata, path, node->name);
- }
- }
- return (result != SDL_ENUM_FAILURE);
- }
- bool Android_JNI_GetAssetPathInfo(const char *path, SDL_PathInfo *info)
- {
- SDL_assert(path != NULL);
- if (!PrepareAPK()) {
- return false;
- }
- path = GetAssetPath(path);
- const APKNode *apknode = FindAPKNode(path);
- if (!apknode) {
- return SDL_SetError("No such file or directory");
- }
- SDL_copyp(info, &apknode->info);
- return true;
- }
- bool Android_JNI_SetClipboardText(const char *text)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- jstring string = (*env)->NewStringUTF(env, text);
- (*env)->CallStaticVoidMethod(env, mActivityClass, midClipboardSetText, string);
- (*env)->DeleteLocalRef(env, string);
- return true;
- }
- char *Android_JNI_GetClipboardText(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- char *text = NULL;
- jstring string;
- string = (*env)->CallStaticObjectMethod(env, mActivityClass, midClipboardGetText);
- if (string) {
- const char *utf = (*env)->GetStringUTFChars(env, string, 0);
- if (utf) {
- text = SDL_strdup(utf);
- (*env)->ReleaseStringUTFChars(env, string, utf);
- }
- (*env)->DeleteLocalRef(env, string);
- }
- return (!text) ? SDL_strdup("") : text;
- }
- bool Android_JNI_HasClipboardText(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midClipboardHasText);
- }
- /* returns 0 on success or -1 on error (others undefined then)
- * returns truthy or falsy value in plugged, charged and battery
- * returns the value in seconds and percent or -1 if not available
- */
- int Android_JNI_GetPowerInfo(int *plugged, int *charged, int *battery, int *seconds, int *percent)
- {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- JNIEnv *env = Android_JNI_GetEnv();
- jmethodID mid;
- jobject context;
- jstring action;
- jclass cls;
- jobject filter;
- jobject intent;
- jstring iname;
- jmethodID imid;
- jstring bname;
- jmethodID bmid;
- if (!LocalReferenceHolder_Init(&refs, env)) {
- LocalReferenceHolder_Cleanup(&refs);
- return -1;
- }
- // context = SDLActivity.getContext();
- context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- action = (*env)->NewStringUTF(env, "android.intent.action.BATTERY_CHANGED");
- cls = (*env)->FindClass(env, "android/content/IntentFilter");
- mid = (*env)->GetMethodID(env, cls, "<init>", "(Ljava/lang/String;)V");
- filter = (*env)->NewObject(env, cls, mid, action);
- (*env)->DeleteLocalRef(env, action);
- mid = (*env)->GetMethodID(env, mActivityClass, "registerReceiver", "(Landroid/content/BroadcastReceiver;Landroid/content/IntentFilter;)Landroid/content/Intent;");
- intent = (*env)->CallObjectMethod(env, context, mid, NULL, filter);
- (*env)->DeleteLocalRef(env, filter);
- cls = (*env)->GetObjectClass(env, intent);
- imid = (*env)->GetMethodID(env, cls, "getIntExtra", "(Ljava/lang/String;I)I");
- // Watch out for C89 scoping rules because of the macro
- #define GET_INT_EXTRA(var, key) \
- int var; \
- iname = (*env)->NewStringUTF(env, key); \
- (var) = (*env)->CallIntMethod(env, intent, imid, iname, -1); \
- (*env)->DeleteLocalRef(env, iname);
- bmid = (*env)->GetMethodID(env, cls, "getBooleanExtra", "(Ljava/lang/String;Z)Z");
- // Watch out for C89 scoping rules because of the macro
- #define GET_BOOL_EXTRA(var, key) \
- int var; \
- bname = (*env)->NewStringUTF(env, key); \
- (var) = (*env)->CallBooleanMethod(env, intent, bmid, bname, JNI_FALSE); \
- (*env)->DeleteLocalRef(env, bname);
- if (plugged) {
- // Watch out for C89 scoping rules because of the macro
- GET_INT_EXTRA(plug, "plugged") // == BatteryManager.EXTRA_PLUGGED (API 5)
- if (plug == -1) {
- LocalReferenceHolder_Cleanup(&refs);
- return -1;
- }
- // 1 == BatteryManager.BATTERY_PLUGGED_AC
- // 2 == BatteryManager.BATTERY_PLUGGED_USB
- *plugged = (0 < plug) ? 1 : 0;
- }
- if (charged) {
- // Watch out for C89 scoping rules because of the macro
- GET_INT_EXTRA(status, "status") // == BatteryManager.EXTRA_STATUS (API 5)
- if (status == -1) {
- LocalReferenceHolder_Cleanup(&refs);
- return -1;
- }
- // 5 == BatteryManager.BATTERY_STATUS_FULL
- *charged = (status == 5) ? 1 : 0;
- }
- if (battery) {
- GET_BOOL_EXTRA(present, "present") // == BatteryManager.EXTRA_PRESENT (API 5)
- *battery = present ? 1 : 0;
- }
- if (seconds) {
- *seconds = -1; // not possible
- }
- if (percent) {
- int level;
- int scale;
- // Watch out for C89 scoping rules because of the macro
- {
- GET_INT_EXTRA(level_temp, "level") // == BatteryManager.EXTRA_LEVEL (API 5)
- level = level_temp;
- }
- // Watch out for C89 scoping rules because of the macro
- {
- GET_INT_EXTRA(scale_temp, "scale") // == BatteryManager.EXTRA_SCALE (API 5)
- scale = scale_temp;
- }
- if ((level == -1) || (scale == -1)) {
- LocalReferenceHolder_Cleanup(&refs);
- return -1;
- }
- *percent = level * 100 / scale;
- }
- (*env)->DeleteLocalRef(env, intent);
- LocalReferenceHolder_Cleanup(&refs);
- return 0;
- }
- // Add all touch devices
- void Android_JNI_InitTouch(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mActivityClass, midInitTouch);
- }
- void Android_JNI_PollInputDevices(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midPollInputDevices);
- }
- void Android_JNI_JoystickSetLED(int device_id, int red, int green, int blue)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midJoystickSetLED, device_id, red, green, blue);
- }
- void Android_JNI_JoystickSetSensorsEnabled(int device_id, bool enabled)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midJoystickSetSensorsEnabled, device_id, (enabled == 1));
- }
- void Android_JNI_PollHapticDevices(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midPollHapticDevices);
- }
- void Android_JNI_HapticRun(int device_id, float intensity, int length)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midHapticRun, device_id, intensity, length);
- }
- void Android_JNI_HapticRumble(int device_id, float low_frequency_intensity, float high_frequency_intensity, int length)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midHapticRumble, device_id, low_frequency_intensity, high_frequency_intensity, length);
- }
- void Android_JNI_HapticStop(int device_id)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mControllerManagerClass, midHapticStop, device_id);
- }
- // See SDLActivity.java for constants.
- #define COMMAND_SET_KEEP_SCREEN_ON 5
- bool SDL_SendAndroidMessage(Uint32 command, int param)
- {
- CHECK_PARAM(command < 0x8000) {
- return SDL_InvalidParamError("command");
- }
- return Android_JNI_SendMessage(command, param);
- }
- // sends message to be handled on the UI event dispatch thread
- bool Android_JNI_SendMessage(int command, int param)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSendMessage, command, param);
- }
- bool Android_JNI_SuspendScreenSaver(bool suspend)
- {
- return Android_JNI_SendMessage(COMMAND_SET_KEEP_SCREEN_ON, (suspend == false) ? 0 : 1);
- }
- void Android_JNI_ShowScreenKeyboard(int input_type, SDL_Rect *inputRect)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticBooleanMethod(env, mActivityClass, midShowTextInput,
- input_type,
- inputRect->x,
- inputRect->y,
- inputRect->w,
- inputRect->h);
- }
- void Android_JNI_HideScreenKeyboard(void)
- {
- // has to match Activity constant
- const int COMMAND_TEXTEDIT_HIDE = 3;
- Android_JNI_SendMessage(COMMAND_TEXTEDIT_HIDE, 0);
- }
- bool Android_JNI_ShowMessageBox(const SDL_MessageBoxData *messageboxdata, int *buttonID)
- {
- JNIEnv *env;
- jclass clazz;
- jmethodID mid;
- jobject context;
- jstring title;
- jstring message;
- jintArray button_flags;
- jintArray button_ids;
- jobjectArray button_texts;
- jintArray colors;
- jobject text;
- jint temp;
- int i;
- env = Android_JNI_GetEnv();
- // convert parameters
- clazz = (*env)->FindClass(env, "java/lang/String");
- title = (*env)->NewStringUTF(env, messageboxdata->title);
- message = (*env)->NewStringUTF(env, messageboxdata->message);
- button_flags = (*env)->NewIntArray(env, messageboxdata->numbuttons);
- button_ids = (*env)->NewIntArray(env, messageboxdata->numbuttons);
- button_texts = (*env)->NewObjectArray(env, messageboxdata->numbuttons,
- clazz, NULL);
- for (i = 0; i < messageboxdata->numbuttons; ++i) {
- const SDL_MessageBoxButtonData *sdlButton;
- if (messageboxdata->flags & SDL_MESSAGEBOX_BUTTONS_RIGHT_TO_LEFT) {
- sdlButton = &messageboxdata->buttons[messageboxdata->numbuttons - 1 - i];
- } else {
- sdlButton = &messageboxdata->buttons[i];
- }
- temp = sdlButton->flags;
- (*env)->SetIntArrayRegion(env, button_flags, i, 1, &temp);
- temp = sdlButton->buttonID;
- (*env)->SetIntArrayRegion(env, button_ids, i, 1, &temp);
- text = (*env)->NewStringUTF(env, sdlButton->text);
- (*env)->SetObjectArrayElement(env, button_texts, i, text);
- (*env)->DeleteLocalRef(env, text);
- }
- if (messageboxdata->colorScheme) {
- colors = (*env)->NewIntArray(env, SDL_MESSAGEBOX_COLOR_COUNT);
- for (i = 0; i < SDL_MESSAGEBOX_COLOR_COUNT; ++i) {
- temp = (0xFFU << 24) |
- (messageboxdata->colorScheme->colors[i].r << 16) |
- (messageboxdata->colorScheme->colors[i].g << 8) |
- (messageboxdata->colorScheme->colors[i].b << 0);
- (*env)->SetIntArrayRegion(env, colors, i, 1, &temp);
- }
- } else {
- colors = NULL;
- }
- (*env)->DeleteLocalRef(env, clazz);
- // context = SDLActivity.getContext();
- context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- clazz = (*env)->GetObjectClass(env, context);
- mid = (*env)->GetMethodID(env, clazz,
- "messageboxShowMessageBox", "(ILjava/lang/String;Ljava/lang/String;[I[I[Ljava/lang/String;[I)I");
- *buttonID = (*env)->CallIntMethod(env, context, mid,
- (jint)messageboxdata->flags,
- title,
- message,
- button_flags,
- button_ids,
- button_texts,
- colors);
- (*env)->DeleteLocalRef(env, context);
- (*env)->DeleteLocalRef(env, clazz);
- // delete parameters
- (*env)->DeleteLocalRef(env, title);
- (*env)->DeleteLocalRef(env, message);
- (*env)->DeleteLocalRef(env, button_flags);
- (*env)->DeleteLocalRef(env, button_ids);
- (*env)->DeleteLocalRef(env, button_texts);
- (*env)->DeleteLocalRef(env, colors);
- return true;
- }
- /*
- //////////////////////////////////////////////////////////////////////////////
- //
- // Functions exposed to SDL applications in SDL_system.h
- //////////////////////////////////////////////////////////////////////////////
- */
- void *SDL_GetAndroidJNIEnv(void)
- {
- return Android_JNI_GetEnv();
- }
- void *SDL_GetAndroidActivity(void)
- {
- // See SDL_system.h for caveats on using this function.
- JNIEnv *env = Android_JNI_GetEnv();
- if (!env) {
- return NULL;
- }
- // return SDLActivity.getContext();
- return (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- }
- int SDL_GetAndroidSDKVersion(void)
- {
- static int sdk_version;
- if (!sdk_version) {
- char sdk[PROP_VALUE_MAX] = { 0 };
- if (__system_property_get("ro.build.version.sdk", sdk) != 0) {
- sdk_version = SDL_atoi(sdk);
- }
- }
- return sdk_version;
- }
- bool SDL_IsAndroidTablet(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsTablet);
- }
- bool SDL_IsAndroidTV(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsAndroidTV);
- }
- bool SDL_IsChromebook(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsChromebook);
- }
- bool SDL_IsDeXMode(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midIsDeXMode);
- }
- void SDL_SendAndroidBackButton(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mActivityClass, midManualBackButton);
- }
- const char *SDL_GetAndroidInternalStoragePath(void)
- {
- static char *s_AndroidInternalFilesPath = NULL;
- if (!s_AndroidInternalFilesPath) {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- jmethodID mid;
- jobject context;
- jobject fileObject;
- jstring pathString;
- const char *path;
- JNIEnv *env = Android_JNI_GetEnv();
- if (!LocalReferenceHolder_Init(&refs, env)) {
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // context = SDLActivity.getContext();
- context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- if (!context) {
- SDL_SetError("Couldn't get Android context!");
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // fileObj = context.getFilesDir();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
- "getFilesDir", "()Ljava/io/File;");
- fileObject = (*env)->CallObjectMethod(env, context, mid);
- if (!fileObject) {
- SDL_SetError("Couldn't get internal directory");
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // path = fileObject.getCanonicalPath();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, fileObject),
- "getCanonicalPath", "()Ljava/lang/String;");
- pathString = (jstring)(*env)->CallObjectMethod(env, fileObject, mid);
- if (Android_JNI_ExceptionOccurred(false)) {
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- path = (*env)->GetStringUTFChars(env, pathString, NULL);
- s_AndroidInternalFilesPath = SDL_strdup(path);
- (*env)->ReleaseStringUTFChars(env, pathString, path);
- LocalReferenceHolder_Cleanup(&refs);
- }
- return s_AndroidInternalFilesPath;
- }
- Uint32 SDL_GetAndroidExternalStorageState(void)
- {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- jmethodID mid;
- jclass cls;
- jstring stateString;
- const char *state_string;
- Uint32 stateFlags;
- JNIEnv *env = Android_JNI_GetEnv();
- if (!LocalReferenceHolder_Init(&refs, env)) {
- LocalReferenceHolder_Cleanup(&refs);
- return 0;
- }
- cls = (*env)->FindClass(env, "android/os/Environment");
- mid = (*env)->GetStaticMethodID(env, cls,
- "getExternalStorageState", "()Ljava/lang/String;");
- stateString = (jstring)(*env)->CallStaticObjectMethod(env, cls, mid);
- state_string = (*env)->GetStringUTFChars(env, stateString, NULL);
- // Print an info message so people debugging know the storage state
- __android_log_print(ANDROID_LOG_INFO, "SDL", "external storage state: %s", state_string);
- if (SDL_strcmp(state_string, "mounted") == 0) {
- stateFlags = SDL_ANDROID_EXTERNAL_STORAGE_READ |
- SDL_ANDROID_EXTERNAL_STORAGE_WRITE;
- } else if (SDL_strcmp(state_string, "mounted_ro") == 0) {
- stateFlags = SDL_ANDROID_EXTERNAL_STORAGE_READ;
- } else {
- stateFlags = 0;
- }
- (*env)->ReleaseStringUTFChars(env, stateString, state_string);
- LocalReferenceHolder_Cleanup(&refs);
- return stateFlags;
- }
- const char *SDL_GetAndroidExternalStoragePath(void)
- {
- static char *s_AndroidExternalFilesPath = NULL;
- if (!s_AndroidExternalFilesPath) {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- jmethodID mid;
- jobject context;
- jobject fileObject;
- jstring pathString;
- const char *path;
- JNIEnv *env = Android_JNI_GetEnv();
- if (!LocalReferenceHolder_Init(&refs, env)) {
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // context = SDLActivity.getContext();
- context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- // fileObj = context.getExternalFilesDir();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
- "getExternalFilesDir", "(Ljava/lang/String;)Ljava/io/File;");
- fileObject = (*env)->CallObjectMethod(env, context, mid, NULL);
- if (!fileObject) {
- SDL_SetError("Couldn't get external directory");
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // path = fileObject.getAbsolutePath();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, fileObject),
- "getAbsolutePath", "()Ljava/lang/String;");
- pathString = (jstring)(*env)->CallObjectMethod(env, fileObject, mid);
- path = (*env)->GetStringUTFChars(env, pathString, NULL);
- s_AndroidExternalFilesPath = SDL_strdup(path);
- (*env)->ReleaseStringUTFChars(env, pathString, path);
- LocalReferenceHolder_Cleanup(&refs);
- }
- return s_AndroidExternalFilesPath;
- }
- const char *SDL_GetAndroidCachePath(void)
- {
- // !!! FIXME: lots of duplication with SDL_GetAndroidExternalStoragePath and SDL_GetAndroidInternalStoragePath; consolidate these functions!
- static char *s_AndroidCachePath = NULL;
- if (!s_AndroidCachePath) {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(SDL_FUNCTION);
- jmethodID mid;
- jobject context;
- jobject fileObject;
- jstring pathString;
- const char *path;
- JNIEnv *env = Android_JNI_GetEnv();
- if (!LocalReferenceHolder_Init(&refs, env)) {
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // context = SDLActivity.getContext();
- context = (*env)->CallStaticObjectMethod(env, mActivityClass, midGetContext);
- // fileObj = context.getExternalFilesDir();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, context),
- "getCacheDir", "()Ljava/io/File;");
- fileObject = (*env)->CallObjectMethod(env, context, mid);
- if (!fileObject) {
- SDL_SetError("Couldn't get cache directory");
- LocalReferenceHolder_Cleanup(&refs);
- return NULL;
- }
- // path = fileObject.getAbsolutePath();
- mid = (*env)->GetMethodID(env, (*env)->GetObjectClass(env, fileObject),
- "getAbsolutePath", "()Ljava/lang/String;");
- pathString = (jstring)(*env)->CallObjectMethod(env, fileObject, mid);
- path = (*env)->GetStringUTFChars(env, pathString, NULL);
- s_AndroidCachePath = SDL_strdup(path);
- (*env)->ReleaseStringUTFChars(env, pathString, path);
- LocalReferenceHolder_Cleanup(&refs);
- }
- return s_AndroidCachePath;
- }
- bool SDL_ShowAndroidToast(const char *message, int duration, int gravity, int xOffset, int yOffset)
- {
- return Android_JNI_ShowToast(message, duration, gravity, xOffset, yOffset);
- }
- void Android_JNI_GetManifestEnvironmentVariables(void)
- {
- if (!mActivityClass || !midGetManifestEnvironmentVariables) {
- __android_log_print(ANDROID_LOG_WARN, "SDL", "Request to get environment variables before JNI is ready");
- return;
- }
- if (!bHasEnvironmentVariables) {
- JNIEnv *env = Android_JNI_GetEnv();
- bool ret = (*env)->CallStaticBooleanMethod(env, mActivityClass, midGetManifestEnvironmentVariables);
- if (ret) {
- bHasEnvironmentVariables = true;
- }
- }
- }
- int Android_JNI_CreateCustomCursor(SDL_Surface *surface, int hot_x, int hot_y)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- int custom_cursor = 0;
- jintArray pixels;
- pixels = (*env)->NewIntArray(env, surface->w * surface->h);
- if (pixels) {
- (*env)->SetIntArrayRegion(env, pixels, 0, surface->w * surface->h, (int *)surface->pixels);
- custom_cursor = (*env)->CallStaticIntMethod(env, mActivityClass, midCreateCustomCursor, pixels, surface->w, surface->h, hot_x, hot_y);
- (*env)->DeleteLocalRef(env, pixels);
- } else {
- SDL_OutOfMemory();
- }
- return custom_cursor;
- }
- void Android_JNI_DestroyCustomCursor(int cursorID)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- (*env)->CallStaticVoidMethod(env, mActivityClass, midDestroyCustomCursor, cursorID);
- }
- bool Android_JNI_SetCustomCursor(int cursorID)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetCustomCursor, cursorID);
- }
- bool Android_JNI_SetSystemCursor(int cursorID)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetSystemCursor, cursorID);
- }
- bool Android_JNI_SupportsRelativeMouse(void)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSupportsRelativeMouse);
- }
- bool Android_JNI_SetRelativeMouseEnabled(bool enabled)
- {
- JNIEnv *env = Android_JNI_GetEnv();
- return (*env)->CallStaticBooleanMethod(env, mActivityClass, midSetRelativeMouseEnabled, (enabled == 1));
- }
- typedef struct NativePermissionRequestInfo
- {
- int request_code;
- char *permission;
- SDL_RequestAndroidPermissionCallback callback;
- void *userdata;
- struct NativePermissionRequestInfo *next;
- } NativePermissionRequestInfo;
- static NativePermissionRequestInfo pending_permissions;
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(nativePermissionResult)(
- JNIEnv *env, jclass cls,
- jint requestCode, jboolean result)
- {
- SDL_LockMutex(Android_ActivityMutex);
- NativePermissionRequestInfo *prev = &pending_permissions;
- for (NativePermissionRequestInfo *info = prev->next; info != NULL; info = info->next) {
- if (info->request_code == (int) requestCode) {
- prev->next = info->next;
- SDL_UnlockMutex(Android_ActivityMutex);
- info->callback(info->userdata, info->permission, result ? true : false);
- SDL_free(info->permission);
- SDL_free(info);
- return;
- }
- prev = info;
- }
- SDL_UnlockMutex(Android_ActivityMutex);
- }
- bool SDL_RequestAndroidPermission(const char *permission, SDL_RequestAndroidPermissionCallback cb, void *userdata)
- {
- if (!permission) {
- return SDL_InvalidParamError("permission");
- } else if (!cb) {
- return SDL_InvalidParamError("cb");
- }
- NativePermissionRequestInfo *info = (NativePermissionRequestInfo *) SDL_calloc(1, sizeof (NativePermissionRequestInfo));
- if (!info) {
- return false;
- }
- info->permission = SDL_strdup(permission);
- if (!info->permission) {
- SDL_free(info);
- return false;
- }
- static SDL_AtomicInt next_request_code;
- info->request_code = SDL_AddAtomicInt(&next_request_code, 1);
- info->callback = cb;
- info->userdata = userdata;
- SDL_LockMutex(Android_ActivityMutex);
- info->next = pending_permissions.next;
- pending_permissions.next = info;
- SDL_UnlockMutex(Android_ActivityMutex);
- JNIEnv *env = Android_JNI_GetEnv();
- jstring jpermission = (*env)->NewStringUTF(env, permission);
- (*env)->CallStaticVoidMethod(env, mActivityClass, midRequestPermission, jpermission, info->request_code);
- (*env)->DeleteLocalRef(env, jpermission);
- return true;
- }
- // Show toast notification
- bool Android_JNI_ShowToast(const char *message, int duration, int gravity, int xOffset, int yOffset)
- {
- bool result;
- JNIEnv *env = Android_JNI_GetEnv();
- jstring jmessage = (*env)->NewStringUTF(env, message);
- result = (*env)->CallStaticBooleanMethod(env, mActivityClass, midShowToast, jmessage, duration, gravity, xOffset, yOffset);
- (*env)->DeleteLocalRef(env, jmessage);
- return result;
- }
- bool Android_JNI_GetLocale(char *buf, size_t buflen)
- {
- bool result = false;
- if (buf && buflen > 0) {
- *buf = '\0';
- JNIEnv *env = Android_JNI_GetEnv();
- jstring string = (jstring)(*env)->CallStaticObjectMethod(env, mActivityClass, midGetPreferredLocales);
- if (string) {
- const char *utf8string = (*env)->GetStringUTFChars(env, string, NULL);
- if (utf8string) {
- result = true;
- SDL_strlcpy(buf, utf8string, buflen);
- (*env)->ReleaseStringUTFChars(env, string, utf8string);
- }
- (*env)->DeleteLocalRef(env, string);
- }
- }
- return result;
- }
- bool Android_JNI_OpenURL(const char *url)
- {
- bool result;
- JNIEnv *env = Android_JNI_GetEnv();
- jstring jurl = (*env)->NewStringUTF(env, url);
- result = (*env)->CallStaticBooleanMethod(env, mActivityClass, midOpenURL, jurl);
- (*env)->DeleteLocalRef(env, jurl);
- return result;
- }
- int Android_JNI_OpenFileDescriptor(const char *uri, const char *mode)
- {
- // Get fopen-style modes
- int moderead = 0, modewrite = 0, modeappend = 0, modeupdate = 0;
- for (const char *cmode = mode; *cmode; cmode++) {
- switch (*cmode) {
- case 'a':
- modeappend = 1;
- break;
- case 'r':
- moderead = 1;
- break;
- case 'w':
- modewrite = 1;
- break;
- case '+':
- modeupdate = 1;
- break;
- default:
- break;
- }
- }
- // Translate fopen-style modes to ContentResolver modes.
- // Android only allows "r", "w", "wt", "wa", "rw" or "rwt".
- const char *contentResolverMode = "r";
- if (moderead) {
- if (modewrite) {
- contentResolverMode = "rwt";
- } else {
- contentResolverMode = modeupdate ? "rw" : "r";
- }
- } else if (modewrite) {
- contentResolverMode = modeupdate ? "rwt" : "wt";
- } else if (modeappend) {
- contentResolverMode = modeupdate ? "rw" : "wa";
- }
- JNIEnv *env = Android_JNI_GetEnv();
- jstring jstringUri = (*env)->NewStringUTF(env, uri);
- jstring jstringMode = (*env)->NewStringUTF(env, contentResolverMode);
- jint fd = (*env)->CallStaticIntMethod(env, mActivityClass, midOpenFileDescriptor, jstringUri, jstringMode);
- (*env)->DeleteLocalRef(env, jstringUri);
- (*env)->DeleteLocalRef(env, jstringMode);
- if (fd == -1) {
- SDL_SetError("Unspecified error in JNI");
- }
- return fd;
- }
- static struct AndroidFileDialog
- {
- int request_code;
- SDL_DialogFileCallback callback;
- void *userdata;
- } mAndroidFileDialogData;
- JNIEXPORT void JNICALL SDL_JAVA_INTERFACE(onNativeFileDialog)(
- JNIEnv *env, jclass jcls,
- jint requestCode, jobjectArray fileList, jint filter)
- {
- if (mAndroidFileDialogData.callback != NULL && mAndroidFileDialogData.request_code == requestCode) {
- if (fileList == NULL) {
- SDL_SetError("Unspecified error in JNI");
- mAndroidFileDialogData.callback(mAndroidFileDialogData.userdata, NULL, -1);
- mAndroidFileDialogData.callback = NULL;
- return;
- }
- // Convert fileList to string
- size_t count = (*env)->GetArrayLength(env, fileList);
- char **charFileList = SDL_calloc(count + 1, sizeof(char *));
- if (charFileList == NULL) {
- mAndroidFileDialogData.callback(mAndroidFileDialogData.userdata, NULL, -1);
- mAndroidFileDialogData.callback = NULL;
- return;
- }
- // Convert to UTF-8
- // TODO: Fix modified UTF-8 to classic UTF-8
- for (int i = 0; i < count; i++) {
- jstring string = (*env)->GetObjectArrayElement(env, fileList, i);
- if (!string) {
- continue;
- }
- const char *utf8string = (*env)->GetStringUTFChars(env, string, NULL);
- if (!utf8string) {
- (*env)->DeleteLocalRef(env, string);
- continue;
- }
- char *newFile = SDL_strdup(utf8string);
- if (!newFile) {
- (*env)->ReleaseStringUTFChars(env, string, utf8string);
- (*env)->DeleteLocalRef(env, string);
- mAndroidFileDialogData.callback(mAndroidFileDialogData.userdata, NULL, -1);
- mAndroidFileDialogData.callback = NULL;
- // Cleanup memory
- for (int j = 0; j < i; j++) {
- SDL_free(charFileList[j]);
- }
- SDL_free(charFileList);
- return;
- }
- charFileList[i] = newFile;
- (*env)->ReleaseStringUTFChars(env, string, utf8string);
- (*env)->DeleteLocalRef(env, string);
- }
- // Call user-provided callback
- SDL_ClearError();
- mAndroidFileDialogData.callback(mAndroidFileDialogData.userdata, (const char *const *) charFileList, filter);
- mAndroidFileDialogData.callback = NULL;
- // Cleanup memory
- for (int i = 0; i < count; i++) {
- SDL_free(charFileList[i]);
- }
- SDL_free(charFileList);
- }
- }
- bool Android_JNI_OpenFileDialog(
- SDL_DialogFileCallback callback, void *userdata,
- const SDL_DialogFileFilter *filters, int nfilters, bool forwrite,
- bool multiple)
- {
- if (mAndroidFileDialogData.callback != NULL) {
- SDL_SetError("Only one file dialog can be run at a time.");
- return false;
- }
- if (forwrite) {
- multiple = false;
- }
- JNIEnv *env = Android_JNI_GetEnv();
- // Setup filters
- jobjectArray filtersArray = NULL;
- if (filters) {
- jclass stringClass = (*env)->FindClass(env, "java/lang/String");
- filtersArray = (*env)->NewObjectArray(env, nfilters, stringClass, NULL);
- (*env)->DeleteLocalRef(env, stringClass);
- // Convert to string
- for (int i = 0; i < nfilters; i++) {
- jstring str = (*env)->NewStringUTF(env, filters[i].pattern);
- (*env)->SetObjectArrayElement(env, filtersArray, i, str);
- (*env)->DeleteLocalRef(env, str);
- }
- }
- // Setup data
- static SDL_AtomicInt next_request_code;
- mAndroidFileDialogData.request_code = SDL_AddAtomicInt(&next_request_code, 1);
- mAndroidFileDialogData.userdata = userdata;
- mAndroidFileDialogData.callback = callback;
- // Invoke JNI
- jboolean success = (*env)->CallStaticBooleanMethod(env, mActivityClass,
- midShowFileDialog, filtersArray, (jboolean) multiple, (jboolean) forwrite, mAndroidFileDialogData.request_code);
- (*env)->DeleteLocalRef(env, filtersArray);
- if (!success) {
- mAndroidFileDialogData.callback = NULL;
- SDL_AddAtomicInt(&next_request_code, -1);
- SDL_SetError("Unspecified error in JNI");
- return false;
- }
- return true;
- }
- #endif // SDL_PLATFORM_ANDROID
|