/* -*- compile-command: "find-and-gradle.sh inXw4dDeb"; -*- */ /* * Copyright 2001-2023 by Eric House (xwords@eehouse.org). All rights * reserved. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "drawwrapper.h" #include "andutils.h" #include "anddict.h" #include "paths.h" enum { JCACHE_RECT0 ,JCACHE_RECT1 #ifdef XWFEATURE_SCOREONEPASS ,JCACHE_DSIS ,JCACHE_RECTS #else ,JCACHE_DSI #endif ,JCACHE_COUNT }; typedef struct _AndDraw { DrawCtxVTable* vtable; #ifdef MAP_THREAD_TO_ENV EnvThreadInfo* ti; #endif jobject jdraw; /* global ref; free it! */ XP_UCHAR curISOCode[MAX_ISO_CODE_LEN+1]; jobject jCache[JCACHE_COUNT]; jobject jTvType; XP_UCHAR miniTextBuf[128]; MPSLOT } AndDraw; #define CHECKOUT_MARKER ((jobject)-1) static void deleteGlobalRef( JNIEnv* env, jobject jobj ); static jobject makeJRect( AndDraw* draw, JNIEnv* env, int indx, const XP_Rect* rect ) { jobject robj = draw->jCache[indx]; #ifdef DEBUG XP_ASSERT( CHECKOUT_MARKER != robj ); #endif int right = rect->left + rect->width; int bottom = rect->top + rect->height; if ( !robj ) { robj = makeObject(env, "android/graphics/Rect", "(IIII)V", rect->left, rect->top, right, bottom ); draw->jCache[indx] = (*env)->NewGlobalRef( env, robj ); deleteLocalRef( env, robj ); robj = draw->jCache[indx]; } else { setInt( env, robj, "left", rect->left ); setInt( env, robj, "top", rect->top ); setInt( env, robj, "right", right ); setInt( env, robj, "bottom", bottom ); } #ifdef DEBUG draw->jCache[indx] = CHECKOUT_MARKER; #endif return robj; } /* makeJRect */ #ifdef DEBUG static void returnJRect( AndDraw* draw, int indx, jobject used ) { XP_ASSERT( CHECKOUT_MARKER == draw->jCache[indx] ); draw->jCache[indx] = used; } #else # define returnJRect( draw, indx, used ) #endif #ifdef XWFEATURE_SCOREONEPASS static void readJRect( JNIEnv* env, XP_Rect* rect, jobject jrect ) { rect->left = getInt( env, jrect, "left" ); rect->top = getInt( env, jrect, "top" ); rect->width = getInt( env, jrect, "right" ) - rect->left; rect->height = getInt( env, jrect, "bottom" ) - rect->top; } static jobject makeJRects( AndDraw* draw, XWEnv xwe, int indx, XP_U16 nPlayers, const XP_Rect rects[] ) { JNIEnv* env = xwe; jobject jrects = draw->jCache[indx]; if ( !jrects ) { jclass rclass = (*env)->FindClass( env, "android/graphics/Rect"); jrects = (*env)->NewObjectArray( env, nPlayers, rclass, NULL ); draw->jCache[indx] = (*env)->NewGlobalRef( env, jrects ); jrects = draw->jCache[indx]; jmethodID initId = (*env)->GetMethodID( env, rclass, "", "()V" ); for ( int ii = 0; ii < nPlayers; ++ii ) { jobject jrect = (*env)->NewObject( env, rclass, initId ); (*env)->SetObjectArrayElement( env, jrects, ii, jrect ); deleteLocalRef( env, jrect ); } deleteLocalRefs( env, rclass, jrects, DELETE_NO_REF ); } if ( NULL != rects ) { XP_ASSERT(0); /* for ( ii = 0; ii < nPlayers; ++ii ) { */ /* jobject jrect = (*env)->GetObjectArrayElement( env, jrects, ii ); */ /* writeJRect( env, jrect, &rects[ii] ); */ /* } */ } return jrects; } static jobject makeDSIs( AndDraw* draw, XWEnv xwe, int indx, XP_U16 nPlayers, const DrawScoreInfo dsis[] ) { JNIEnv* env = xwe; jobject dsiobjs = draw->jCache[indx]; if ( !dsiobjs ) { jclass clas = (*env)->FindClass( env, PKG_PATH("jni/DrawScoreInfo") ); dsiobjs = (*env)->NewObjectArray( env, nPlayers, clas, NULL ); draw->jCache[indx] = (*env)->NewGlobalRef( env, dsiobjs ); deleteLocalRef( env, dsiobjs ); dsiobjs = draw->jCache[indx]; jmethodID initId = (*env)->GetMethodID( env, clas, "", "()V" ); for ( int ii = 0; ii < nPlayers; ++ii ) { jobject dsiobj = (*env)->NewObject( env, clas, initId ); (*env)->SetObjectArrayElement( env, dsiobjs, ii, dsiobj ); deleteLocalRef( env, dsiobj ); } deleteLocalRef( env, clas ); } for ( int ii = 0; ii < nPlayers; ++ii ) { jobject dsiobj = (*env)->GetObjectArrayElement( env, dsiobjs, ii ); const DrawScoreInfo* dsi = &dsis[ii]; setInt( env, dsiobj, "playerNum", dsi->playerNum ); setInt( env, dsiobj, "totalScore", dsi->totalScore ); setInt( env, dsiobj, "nTilesLeft", dsi->nTilesLeft ); setInt( env, dsiobj, "flags", dsi->flags ); setBool( env, dsiobj, "isTurn", dsi->isTurn ); setBool( env, dsiobj, "selected", dsi->selected ); setBool( env, dsiobj, "isRemote", dsi->isRemote ); setBool( env, dsiobj, "isRobot", dsi->isRobot ); setString( env, dsiobj, "name", dsi->name ); } return dsiobjs; } #else static jobject makeDSI( AndDraw* draw, XWEnv xwe, int indx, const DrawScoreInfo* dsi ) { JNIEnv* env = xwe; jobject dsiobj = draw->jCache[indx]; if ( !dsiobj ) { dsiobj = makeObjectEmptyConstr( env, PKG_PATH("jni/DrawScoreInfo") ); draw->jCache[indx] = (*env)->NewGlobalRef( env, dsiobj ); deleteLocalRef( env, dsiobj ); dsiobj = draw->jCache[indx]; } setInt( env, dsiobj, "playerNum", dsi->playerNum ); setInt( env, dsiobj, "totalScore", dsi->totalScore ); setInt( env, dsiobj, "nTilesLeft", dsi->nTilesLeft ); setInt( env, dsiobj, "flags", dsi->flags ); setBool( env, dsiobj, "isTurn", dsi->isTurn ); setBool( env, dsiobj, "selected", dsi->selected ); setBool( env, dsiobj, "isRemote", dsi->isRemote ); setBool( env, dsiobj, "isRobot", dsi->isRobot ); setString( env, dsiobj, "name", dsi->name ); return dsiobj; } #endif #define DRAW_CBK_HEADER(nam,sig) { \ JNIEnv* env = xwe; \ AndDraw* draw = (AndDraw*)dctx; \ ASSERT_ENV( draw->ti, env ); \ XP_ASSERT( !!draw->jdraw ); \ jmethodID mid = getMethodID( xwe, draw->jdraw, nam, sig ) \ #define DRAW_CBK_HEADER_END() } static XP_Bool and_draw_scoreBegin( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, XP_U16 numPlayers, const XP_S16* const scores, XP_S16 remCount, DrawFocusState XP_UNUSED(dfs) ) { jboolean result; DRAW_CBK_HEADER("scoreBegin", "(Landroid/graphics/Rect;I[II)Z" ); jint jarr[numPlayers]; for ( int ii = 0; ii < numPlayers; ++ii ) { jarr[ii] = scores[ii]; } jintArray jscores = makeIntArray( env, numPlayers, jarr, sizeof(jarr[0]) ); jobject jrect = makeJRect( draw, env, JCACHE_RECT0, rect ); result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, numPlayers, jscores, remCount ); returnJRect( draw, JCACHE_RECT0, jrect ); deleteLocalRef( env, jscores ); DRAW_CBK_HEADER_END(); return result; } #ifdef XWFEATURE_SCOREONEPASS static XP_Bool and_draw_drawRemText( DrawCtx* dctx, XP_S16 nTilesLeft, XP_Bool focussed, XP_Rect* rect ) { DRAW_CBK_HEADER("drawRemText", "(IZLandroid/graphics/Rect;)Z" ); jobject jrect = makeJRect( draw, env, JCACHE_RECT0, rect ); jboolean result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, nTilesLeft, focussed, jrect ); if ( result ) { readJRect( env, rect, jrect ); } returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); return result; } static void and_draw_score_drawPlayers( DrawCtx* dctx, XWEnv xwe, const XP_Rect* scoreRect, XP_U16 nPlayers, DrawScoreInfo playerData[], XP_Rect playerRects[] ) { DRAW_CBK_HEADER("score_drawPlayers", "(Landroid/graphics/Rect;" "[L" PKG_PATH("jni/DrawScoreInfo;") "[Landroid/graphics/Rect;)V" ); jobject jrect = makeJRect( draw, env, JCACHE_RECT0, scoreRect ); jobject jdsis = makeDSIs( draw, xwe, JCACHE_DSIS, nPlayers, playerData ); jobject jrects = makeJRects( draw, env, JCACHE_RECTS, nPlayers, NULL ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, jdsis, jrects ); for ( int ii = 0; ii < nPlayers; ++ii ) { jobject jrect = (*env)->GetObjectArrayElement( env, jrects, ii ); readJRect( env, &playerRects[ii], jrect ); } returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); } #else static XP_Bool and_draw_measureRemText( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, XP_S16 nTilesLeft, XP_U16* width, XP_U16* height ) { jboolean result; DRAW_CBK_HEADER("measureRemText", "(Landroid/graphics/Rect;I[I[I)Z" ); jintArray widthArray = (*env)->NewIntArray( env, 1 ); jintArray heightArray = (*env)->NewIntArray( env, 1 ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, nTilesLeft, widthArray, heightArray ); if ( result ) { int tmp; getIntsFromArray( env, &tmp, widthArray, 1, true ); *width = tmp; getIntsFromArray( env, &tmp, heightArray, 1, true ); *height = tmp; } returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); return result; } /* and_draw_measureRemText */ static void and_draw_drawRemText( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rInner, const XP_Rect* rOuter, XP_S16 nTilesLeft, XP_Bool focussed ) { DRAW_CBK_HEADER("drawRemText", "(Landroid/graphics/Rect;Landroid/graphics/Rect;IZ)V" ); jobject jrinner = makeJRect( draw, env, JCACHE_RECT0, rInner ); jobject jrouter = makeJRect( draw, env, JCACHE_RECT1, rOuter ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrinner, jrouter, nTilesLeft, focussed ); returnJRect( draw, JCACHE_RECT0, jrinner ); returnJRect( draw, JCACHE_RECT1, jrouter ); DRAW_CBK_HEADER_END(); } static void and_draw_measureScoreText( DrawCtx* dctx, XWEnv xwe, const XP_Rect* r, const DrawScoreInfo* dsi, XP_U16* width, XP_U16* height ) { DRAW_CBK_HEADER("measureScoreText", "(Landroid/graphics/Rect;L" PKG_PATH("jni/DrawScoreInfo;[I[I)V") ); jobject jrect = makeJRect( draw, env, JCACHE_RECT0, r ); jobject jdsi = makeDSI( draw, xwe, JCACHE_DSI, dsi ); jintArray widthArray = (*env)->NewIntArray( env, 1 ); jintArray heightArray = (*env)->NewIntArray( env, 1 ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, jdsi, widthArray, heightArray ); returnJRect( draw, JCACHE_RECT0, jrect ); int tmp; getIntsFromArray( env, &tmp, widthArray, 1, true ); *width = tmp; getIntsFromArray( env, &tmp, heightArray, 1, true ); *height = tmp; DRAW_CBK_HEADER_END(); } /* and_draw_measureScoreText */ static void and_draw_score_drawPlayer( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rInner, const XP_Rect* rOuter, XP_U16 gotPct, const DrawScoreInfo* dsi ) { DRAW_CBK_HEADER("score_drawPlayer", "(Landroid/graphics/Rect;Landroid/graphics/Rect;I" "L" PKG_PATH("jni/DrawScoreInfo") ";)V" ); jobject jrinner = makeJRect( draw, xwe, JCACHE_RECT0, rInner ); jobject jrouter = makeJRect( draw, xwe, JCACHE_RECT1, rOuter ); jobject jdsi = makeDSI( draw, xwe, JCACHE_DSI, dsi ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrinner, jrouter, gotPct, jdsi ); returnJRect( draw, JCACHE_RECT0, jrinner ); returnJRect( draw, JCACHE_RECT1, jrouter ); DRAW_CBK_HEADER_END(); } /* and_draw_score_drawPlayer */ #endif static void and_draw_drawTimer( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, XP_U16 player, XP_S16 secondsLeft, XP_Bool inDuplicateMode ) { if ( rect->width == 0 ) { XP_LOGFF( "exiting b/c rect empty" ); } else { DRAW_CBK_HEADER("drawTimer", "(Landroid/graphics/Rect;IIZ)V" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, player, secondsLeft, inDuplicateMode ); returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); } } /* Not used on android yet */ static XP_Bool and_draw_beginDraw( DrawCtx* XP_UNUSED(dctx), XWEnv XP_UNUSED(xwe) ) { return XP_TRUE; } static void and_draw_endDraw( DrawCtx* XP_UNUSED(dctx), XWEnv XP_UNUSED(xwe) ) {} static XP_Bool and_draw_boardBegin( DrawCtx* dctx, XWEnv xwe, const XP_Rect* XP_UNUSED(rect), XP_U16 XP_UNUSED(cellWidth), XP_U16 XP_UNUSED(cellHeight), DrawFocusState XP_UNUSED(dfs), TileValueType tvType ) { JNIEnv* env = xwe; AndDraw* draw = (AndDraw*)dctx; jobject jTvType = intToJEnum( env, tvType, PKG_PATH("jni/CommonPrefs$TileValueType") ); draw->jTvType = (*env)->NewGlobalRef( env, jTvType ); deleteLocalRef( env, jTvType ); return XP_TRUE; } static XP_Bool and_draw_drawCell( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, const XP_UCHAR* text, const XP_Bitmaps* bitmaps, Tile tile, XP_U16 value, XP_S16 owner, XWBonusType bonus, HintAtts hintAtts, CellFlags flags ) { jboolean result; DRAW_CBK_HEADER("drawCell", "(Landroid/graphics/Rect;Ljava/lang/String;IIIII" "L" PKG_PATH("jni/CommonPrefs$TileValueType") ";)Z" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); jstring jtext = NULL; if ( !!text ) { if ( 0 == strcmp( "_", text ) ) { text = "?"; } jtext = (*env)->NewStringUTF( env, text ); } result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, jtext, tile, value, owner, bonus, flags, draw->jTvType ); returnJRect( draw, JCACHE_RECT0, jrect ); deleteLocalRef( env, jtext ); DRAW_CBK_HEADER_END(); return result; } static void and_draw_drawBoardArrow( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, XWBonusType bonus,XP_Bool vert, HintAtts hintAtts, CellFlags flags ) { DRAW_CBK_HEADER("drawBoardArrow", "(Landroid/graphics/Rect;IZII)V" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, bonus, vert, hintAtts, flags ); returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); } static XP_Bool and_draw_vertScrollBoard( DrawCtx* XP_UNUSED(dctx), XWEnv XP_UNUSED(xwe), XP_Rect* XP_UNUSED(rect), XP_S16 XP_UNUSED(dist), DrawFocusState XP_UNUSED(dfs) ) { /* Scrolling a bitmap in-place isn't any faster than drawing every cell anew so no point in calling into java. */ return XP_FALSE; } static XP_Bool and_draw_trayBegin( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, XP_U16 owner, XP_S16 score, DrawFocusState XP_UNUSED(dfs) ) { jboolean result; DRAW_CBK_HEADER( "trayBegin", "(Landroid/graphics/Rect;II)Z" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, owner, score ); returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); return result; } static XP_Bool and_draw_drawTile( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, const XP_UCHAR* text, const XP_Bitmaps* bitmaps, XP_S16 val, CellFlags flags ) { XP_Bool result; DRAW_CBK_HEADER( "drawTile", "(Landroid/graphics/Rect;Ljava/lang/String;II)Z" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); jstring jtext = NULL; if ( !!text ) { jtext = (*env)->NewStringUTF( env, text ); } result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, jtext, val, flags ); returnJRect( draw, JCACHE_RECT0, jrect ); deleteLocalRef( env, jtext ); DRAW_CBK_HEADER_END(); return result; } static XP_Bool and_draw_drawTileMidDrag( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, const XP_UCHAR* text, const XP_Bitmaps* bitmaps, XP_U16 val, XP_U16 owner, CellFlags flags ) { XP_Bool result; DRAW_CBK_HEADER( "drawTileMidDrag", "(Landroid/graphics/Rect;Ljava/lang/String;III)Z" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); jstring jtext = NULL; if ( !!text ) { jtext = (*env)->NewStringUTF( env, text ); } result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, jtext, val, owner, flags ); returnJRect( draw, JCACHE_RECT0, jrect ); deleteLocalRef( env, jtext ); DRAW_CBK_HEADER_END(); return result; } static XP_Bool and_draw_drawTileBack( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, CellFlags flags ) { XP_Bool result; DRAW_CBK_HEADER( "drawTileBack", "(Landroid/graphics/Rect;I)Z" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, flags ); returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); return result; } static void and_draw_drawTrayDivider( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, CellFlags flags ) { DRAW_CBK_HEADER( "drawTrayDivider", "(Landroid/graphics/Rect;I)V" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, flags ); returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); } static void and_draw_score_pendingScore( DrawCtx* dctx, XWEnv xwe, const XP_Rect* rect, XP_S16 score, XP_U16 playerNum, XP_Bool curTurn, CellFlags flags ) { DRAW_CBK_HEADER( "score_pendingScore", "(Landroid/graphics/Rect;IIZI)V" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, score, playerNum, curTurn, flags ); returnJRect( draw, JCACHE_RECT0, jrect ); DRAW_CBK_HEADER_END(); } static void and_draw_objFinished( DrawCtx* dctx, XWEnv xwe, BoardObjectType typ, const XP_Rect* rect, DrawFocusState XP_UNUSED(dfs) ) { #ifndef XWFEATURE_SCOREONEPASS DRAW_CBK_HEADER( "objFinished", "(ILandroid/graphics/Rect;)V" ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); (*env)->CallVoidMethod( env, draw->jdraw, mid, (jint)typ, jrect ); returnJRect( draw, JCACHE_RECT0, jrect ); if ( OBJ_BOARD == typ ) { deleteGlobalRef( env, draw->jTvType ); } DRAW_CBK_HEADER_END(); #endif } static void and_draw_dictChanged( DrawCtx* dctx, XWEnv xwe, XP_S16 playerNum, const DictionaryCtxt* dict ) { AndDraw* draw = (AndDraw*)dctx; if ( !!dict && !!draw->jdraw ) { XP_LOGFF( "(dict=%p/%s); code=%x", dict, dict_getName(dict), andDictID(dict) ); const XP_UCHAR* isoCode = NULL; /* A null dict means no-lang */ if ( NULL != dict ) { isoCode = dict_getISOCode( dict ); } /* Don't bother sending repeats. */ if ( 0 != XP_STRCMP( isoCode, draw->curISOCode ) ) { XP_STRNCPY( draw->curISOCode, isoCode, VSIZE(draw->curISOCode) ); DRAW_CBK_HEADER( "dictChanged", "(J)V" ); /* /\* create a DictWrapper object -- if the API changes to require it *\/ */ /* jclass rclass = (*env)->FindClass( env, PKG_PATH("jni/XwJNI$DictWrapper") ); */ /* // http://stackoverflow.com/questions/7260376/how-to-create-an-object-with-jni */ /* const char* sig = "(L" PKG_PATH("jni/XwJNI") ";I)V"; */ /* jmethodID initId = (*env)->GetMethodID( env, rclass, "", sig ); */ /* jobject jdict = (*env)->NewObject( env, rclass, initId, (int)dict ); */ (*env)->CallVoidMethod( env, draw->jdraw, mid, (jlong)dict ); DRAW_CBK_HEADER_END(); } } } #ifdef XWFEATURE_MINIWIN static const XP_UCHAR* and_draw_getMiniWText( DrawCtx* dctx, XWEnv xwe, XWMiniTextType textHint ) { DRAW_CBK_HEADER( "getMiniWText", "(I)Ljava/lang/String;" ); jstring jstr = (*env)->CallObjectMethod( env, draw->jdraw, mid, textHint ); const char* str = (*env)->GetStringUTFChars( env, jstr, NULL ); snprintf( draw->miniTextBuf, VSIZE(draw->miniTextBuf), "%s", str ); (*env)->ReleaseStringUTFChars( env, jstr, str ); deleteLocalRef( env, jstr ); DRAW_CBK_HEADER_END(); return draw->miniTextBuf; } static void and_draw_measureMiniWText( DrawCtx* dctx, XWEnv xwe, const XP_UCHAR* textP, XP_U16* width, XP_U16* height ) { DRAW_CBK_HEADER( "measureMiniWText", "(Ljava/lang/String;[I[I)V" ); jintArray widthArray = (*env)->NewIntArray( env, 1 ); jintArray heightArray = (*env)->NewIntArray( env, 1 ); jstring jstr = (*env)->NewStringUTF( env, textP ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jstr, widthArray, heightArray ); deleteLocalRef( env, jstr ); int tmp; getIntsFromArray( env, &tmp, widthArray, 1, true ); *width = tmp; getIntsFromArray( env, &tmp, heightArray, 1, true ); *height = tmp; DRAW_CBK_HEADER_END(); } static void and_draw_drawMiniWindow( DrawCtx* dctx, XWEnv xwe, const XP_UCHAR* text, const XP_Rect* rect, void** closure ) { DRAW_CBK_HEADER( "drawMiniWindow", "(Ljava/lang/String;Landroid/graphics/Rect;)V" ); jstring jstr = (*env)->NewStringUTF( env, text ); jobject jrect = makeJRect( draw, xwe, JCACHE_RECT0, rect ); (*env)->CallVoidMethod( env, draw->jdraw, mid, jstr, jrect ); returnJRect( draw, JCACHE_RECT0, jrect ); deleteLocalRef( env, jstr ); DRAW_CBK_HEADER_END(); } #endif static XP_Bool draw_doNothing( DrawCtx* dctx, XWEnv xwe, ... ) { LOG_FUNC(); return XP_FALSE; } /* draw_doNothing */ DrawCtx* makeDraw( MPFORMAL JNIEnv* env, #ifdef MAP_THREAD_TO_ENV EnvThreadInfo* ti, #endif jobject jdraw ) { AndDraw* draw = (AndDraw*)XP_CALLOC( mpool, sizeof(*draw) ); #ifdef MAP_THREAD_TO_ENV draw->ti = ti; #endif draw->vtable = XP_MALLOC( mpool, sizeof(*draw->vtable) ); if ( NULL != jdraw ) { draw->jdraw = (*env)->NewGlobalRef( env, jdraw ); } MPASSIGN( draw->mpool, mpool ); for ( int ii = 0; ii < sizeof(*draw->vtable)/sizeof(void*); ++ii ) { ((void**)(draw->vtable))[ii] = draw_doNothing; } #define SET_PROC(nam) draw->vtable->m_draw_##nam = and_draw_##nam SET_PROC(beginDraw); SET_PROC(endDraw); SET_PROC(boardBegin); SET_PROC(scoreBegin); #ifdef XWFEATURE_SCOREONEPASS SET_PROC(score_drawPlayers); SET_PROC(drawRemText); #else SET_PROC(measureScoreText); SET_PROC(score_drawPlayer); SET_PROC(measureRemText); SET_PROC(drawRemText); #endif SET_PROC(drawTimer); SET_PROC(drawCell); SET_PROC(drawBoardArrow); SET_PROC(vertScrollBoard); SET_PROC(trayBegin); SET_PROC(drawTile); SET_PROC(drawTileMidDrag); SET_PROC(drawTileBack); SET_PROC(drawTrayDivider); SET_PROC(score_pendingScore); SET_PROC(objFinished); SET_PROC(dictChanged); #ifdef XWFEATURE_MINIWIN SET_PROC(getMiniWText); SET_PROC(measureMiniWText); SET_PROC(drawMiniWindow); #endif #undef SET_PROC return (DrawCtx*)draw; } static void deleteGlobalRef( JNIEnv* env, jobject jobj ) { if ( !!jobj ) { (*env)->DeleteGlobalRef( env, jobj ); } } void destroyDraw( DrawCtx** dctx, JNIEnv* env ) { if ( !!*dctx ) { AndDraw* draw = (AndDraw*)*dctx; deleteGlobalRef( env, draw->jdraw ); for ( int ii = 0; ii < JCACHE_COUNT; ++ii ) { deleteGlobalRef( env, draw->jCache[ii] ); } XP_FREE( draw->mpool, draw->vtable ); XP_FREE( draw->mpool, draw ); *dctx = NULL; } }