| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2025 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>
- #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(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);
- 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) },
- { "nativeAddJoystick", "(ILjava/lang/String;Ljava/lang/String;IIIIIIZZ)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 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");
- 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 || !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(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)
- {
- #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);
- (*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;
- }
- static void Internal_Android_Create_AssetManager(void)
- {
- struct LocalReferenceHolder refs = LocalReferenceHolder_Setup(__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;
- }
- }
- 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 (!asset_manager) {
- Internal_Android_Create_AssetManager();
- if (!asset_manager) {
- return SDL_SetError("Couldn't create asset manager");
- }
- }
- path = GetAssetPath(path);
- AAssetDir *adir = AAssetManager_openDir(asset_manager, path);
- if (!adir) {
- return SDL_SetError("AAssetManager_openDir failed");
- }
- SDL_EnumerationResult result = SDL_ENUM_CONTINUE;
- const char *ent;
- while ((result == SDL_ENUM_CONTINUE) && ((ent = AAssetDir_getNextFileName(adir)) != NULL)) {
- result = cb(userdata, path, ent);
- }
- AAssetDir_close(adir);
- return (result != SDL_ENUM_FAILURE);
- }
- bool Android_JNI_GetAssetPathInfo(const char *path, SDL_PathInfo *info)
- {
- if (!asset_manager) {
- Internal_Android_Create_AssetManager();
- if (!asset_manager) {
- return SDL_SetError("Couldn't create asset manager");
- }
- }
- path = GetAssetPath(path);
- // this is sort of messy, but there isn't a stat()-like interface to the Assets.
- AAsset *aasset = AAssetManager_open(asset_manager, path, AASSET_MODE_UNKNOWN);
- if (aasset) { // it's a file!
- info->type = SDL_PATHTYPE_FILE;
- info->size = (Uint64) AAsset_getLength64(aasset);
- AAsset_close(aasset);
- return true;
- }
- AAssetDir *adir = AAssetManager_openDir(asset_manager, path);
- if (adir) { // This does _not_ return NULL for a missing directory! Treat empty directories as missing. Better than nothing. :/
- const bool contains_something = (AAssetDir_getNextFileName(adir) != NULL); // if not NULL, there are files in this directory, so it's _definitely_ a directory.
- AAssetDir_close(adir);
- if (contains_something) {
- info->type = SDL_PATHTYPE_DIRECTORY;
- info->size = 0;
- return true;
- }
- }
- return SDL_SetError("Couldn't open asset '%s'", path);
- }
- 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(__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_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(__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(__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(__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(__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
|