mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-11 08:01:22 +01:00
e6561cbcac
commented-out logging of drawCell, with flags; debug-build checks that static rects passed to java draw code aren't being used by multiple threads at once.
711 lines
24 KiB
C
711 lines
24 KiB
C
/* -*-mode: C; compile-command: "cd ..; ../scripts/ndkbuild.sh -j3"; -*- */
|
|
/*
|
|
* Copyright 2001-2010 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;
|
|
EnvThreadInfo* ti;
|
|
jobject jdraw; /* global ref; free it! */
|
|
XP_LangCode curLang;
|
|
jobject jCache[JCACHE_COUNT];
|
|
XP_UCHAR miniTextBuf[128];
|
|
MPSLOT
|
|
} AndDraw;
|
|
|
|
#define CHECKOUT_MARKER ((jobject)-1)
|
|
|
|
static jobject
|
|
makeJRect( AndDraw* draw, int indx, const XP_Rect* rect )
|
|
{
|
|
JNIEnv* env = ENVFORME( draw->ti );
|
|
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 ) {
|
|
jclass rclass = (*env)->FindClass( env, "android/graphics/Rect");
|
|
jmethodID initId = (*env)->GetMethodID( env, rclass, "<init>",
|
|
"(IIII)V" );
|
|
robj = (*env)->NewObject( env, rclass, initId, rect->left, rect->top,
|
|
right, bottom );
|
|
|
|
draw->jCache[indx] = (*env)->NewGlobalRef( env, robj );
|
|
deleteLocalRefs( env, robj, rclass, DELETE_NO_REF );
|
|
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, int indx, XP_U16 nPlayers, const XP_Rect rects[] )
|
|
{
|
|
JNIEnv* env = ENVFORME( draw->ti );
|
|
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, "<init>",
|
|
"()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, int indx, XP_U16 nPlayers, const DrawScoreInfo dsis[] )
|
|
{
|
|
JNIEnv* env = ENVFORME( draw->ti );
|
|
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, "<init>", "()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, int indx, const DrawScoreInfo* dsi )
|
|
{
|
|
JNIEnv* env = ENVFORME( draw->ti );
|
|
jobject dsiobj = draw->jCache[indx];
|
|
|
|
if ( !dsiobj ) {
|
|
jclass rclass = (*env)->FindClass( env, PKG_PATH("jni/DrawScoreInfo") );
|
|
jmethodID initId = (*env)->GetMethodID( env, rclass, "<init>", "()V" );
|
|
dsiobj = (*env)->NewObject( env, rclass, initId );
|
|
|
|
draw->jCache[indx] = (*env)->NewGlobalRef( env, dsiobj );
|
|
deleteLocalRefs( env, rclass, dsiobj, DELETE_NO_REF );
|
|
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) \
|
|
AndDraw* draw = (AndDraw*)dctx; \
|
|
JNIEnv* env = ENVFORME( draw->ti ); \
|
|
XP_ASSERT( !!draw->jdraw ); \
|
|
jmethodID mid = getMethodID( env, draw->jdraw, nam, sig );
|
|
|
|
static XP_Bool
|
|
and_draw_scoreBegin( DrawCtx* dctx, 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 );
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
result = (*env)->CallBooleanMethod( env, draw->jdraw, mid,
|
|
jrect, numPlayers, jscores, remCount );
|
|
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
deleteLocalRef( env, jscores );
|
|
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, 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 );
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_draw_score_drawPlayers( DrawCtx* dctx, 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, JCACHE_RECT0, scoreRect );
|
|
jobject jdsis = makeDSIs( draw, JCACHE_DSIS, nPlayers, playerData );
|
|
jobject jrects = makeJRects( draw, 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 );
|
|
}
|
|
|
|
#else
|
|
|
|
static XP_Bool
|
|
and_draw_measureRemText( DrawCtx* dctx, const XP_Rect* rect,
|
|
XP_S16 nTilesLeft,
|
|
XP_U16* width, XP_U16* height )
|
|
{
|
|
DRAW_CBK_HEADER("measureRemText", "(Landroid/graphics/Rect;I[I[I)Z" );
|
|
|
|
jintArray widthArray = makeIntArray( env, 1, NULL );
|
|
jintArray heightArray = makeIntArray( env, 1, NULL );
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
jboolean result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect,
|
|
nTilesLeft, widthArray,
|
|
heightArray );
|
|
if ( result ) {
|
|
*width = getIntFromArray( env, widthArray, true );
|
|
*height = getIntFromArray( env, heightArray, true );
|
|
}
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
return result;
|
|
} /* and_draw_measureRemText */
|
|
|
|
static void
|
|
and_draw_drawRemText( DrawCtx* dctx, 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, JCACHE_RECT0, rInner );
|
|
jobject jrouter = makeJRect( draw, JCACHE_RECT1, rOuter );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid, jrinner, jrouter,
|
|
nTilesLeft, focussed );
|
|
returnJRect( draw, JCACHE_RECT0, jrinner );
|
|
returnJRect( draw, JCACHE_RECT1, jrouter );
|
|
}
|
|
|
|
static void
|
|
and_draw_measureScoreText( DrawCtx* dctx,
|
|
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, JCACHE_RECT0, r );
|
|
jobject jdsi = makeDSI( draw, JCACHE_DSI, dsi );
|
|
|
|
jintArray widthArray = makeIntArray( env, 1, NULL );
|
|
jintArray heightArray = makeIntArray( env, 1, NULL );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid, jrect, jdsi,
|
|
widthArray, heightArray );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
|
|
*width = getIntFromArray( env, widthArray, true );
|
|
*height = getIntFromArray( env, heightArray, true );
|
|
} /* and_draw_measureScoreText */
|
|
|
|
static void
|
|
and_draw_score_drawPlayer( DrawCtx* dctx, 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, JCACHE_RECT0, rInner );
|
|
jobject jrouter = makeJRect( draw, JCACHE_RECT1, rOuter );
|
|
jobject jdsi = makeDSI( draw, JCACHE_DSI, dsi );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid, jrinner, jrouter, gotPct,
|
|
jdsi );
|
|
returnJRect( draw, JCACHE_RECT0, jrinner );
|
|
returnJRect( draw, JCACHE_RECT1, jrouter );
|
|
} /* and_draw_score_drawPlayer */
|
|
#endif
|
|
|
|
static void
|
|
and_draw_drawTimer( DrawCtx* dctx, const XP_Rect* rect, XP_U16 player,
|
|
XP_S16 secondsLeft )
|
|
{
|
|
if ( rect->width == 0 ) {
|
|
XP_LOGF( "%s: exiting b/c rect empty", __func__ );
|
|
} else {
|
|
DRAW_CBK_HEADER("drawTimer", "(Landroid/graphics/Rect;II)V" );
|
|
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
jrect, player, secondsLeft );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
}
|
|
}
|
|
|
|
/* Not used on android yet */
|
|
static XP_Bool and_draw_beginDraw( DrawCtx* XP_UNUSED(dctx) ) {return XP_TRUE;}
|
|
static void and_draw_endDraw( DrawCtx* XP_UNUSED(dctx) ) {}
|
|
|
|
static XP_Bool
|
|
and_draw_boardBegin( DrawCtx* XP_UNUSED(dctx), const XP_Rect* XP_UNUSED(rect),
|
|
XP_U16 XP_UNUSED(cellWidth), XP_U16 XP_UNUSED(cellHeight),
|
|
DrawFocusState XP_UNUSED(dfs) )
|
|
{
|
|
return XP_TRUE;
|
|
}
|
|
|
|
static XP_Bool
|
|
and_draw_drawCell( DrawCtx* dctx, 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 )
|
|
{
|
|
DRAW_CBK_HEADER("drawCell",
|
|
"(Landroid/graphics/Rect;Ljava/lang/String;IIIIII)Z" );
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
jstring jtext = NULL;
|
|
if ( !!text ) {
|
|
if ( 0 == strcmp( "_", text ) ) {
|
|
text = "?";
|
|
}
|
|
jtext = (*env)->NewStringUTF( env, text );
|
|
}
|
|
|
|
jboolean result = (*env)->CallBooleanMethod( env, draw->jdraw, mid,
|
|
jrect, jtext, tile, value,
|
|
owner, bonus, hintAtts,
|
|
flags );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
deleteLocalRef( env, jtext );
|
|
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_draw_drawBoardArrow(DrawCtx* dctx, 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, JCACHE_RECT0, rect );
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
jrect, bonus, vert, hintAtts, flags );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
}
|
|
|
|
static XP_Bool
|
|
and_draw_vertScrollBoard( DrawCtx* XP_UNUSED(dctx), 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, const XP_Rect* rect, XP_U16 owner,
|
|
XP_S16 score, DrawFocusState XP_UNUSED(dfs) )
|
|
{
|
|
DRAW_CBK_HEADER( "trayBegin", "(Landroid/graphics/Rect;II)Z" );
|
|
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
jboolean result = (*env)->CallBooleanMethod( env, draw->jdraw, mid,
|
|
jrect, owner, score );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
return result;
|
|
}
|
|
|
|
static XP_Bool
|
|
and_draw_drawTile( DrawCtx* dctx, const XP_Rect* rect, const XP_UCHAR* text,
|
|
const XP_Bitmaps* bitmaps, XP_U16 val, CellFlags flags )
|
|
{
|
|
XP_Bool result;
|
|
DRAW_CBK_HEADER( "drawTile",
|
|
"(Landroid/graphics/Rect;Ljava/lang/String;II)Z" );
|
|
jobject jrect = makeJRect( draw, 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 );
|
|
return result;
|
|
}
|
|
|
|
static XP_Bool
|
|
and_draw_drawTileMidDrag( DrawCtx* dctx, 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, 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 );
|
|
return result;
|
|
}
|
|
|
|
static XP_Bool
|
|
and_draw_drawTileBack( DrawCtx* dctx, const XP_Rect* rect, CellFlags flags )
|
|
{
|
|
DRAW_CBK_HEADER( "drawTileBack", "(Landroid/graphics/Rect;I)Z" );
|
|
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
XP_Bool result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, flags );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_draw_drawTrayDivider( DrawCtx* dctx, const XP_Rect* rect, CellFlags flags )
|
|
{
|
|
DRAW_CBK_HEADER( "drawTrayDivider", "(Landroid/graphics/Rect;I)V" );
|
|
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
jrect, flags );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
}
|
|
|
|
static void
|
|
and_draw_score_pendingScore( DrawCtx* dctx, const XP_Rect* rect,
|
|
XP_S16 score, XP_U16 playerNum,
|
|
XP_S16 curTurn, CellFlags flags )
|
|
{
|
|
DRAW_CBK_HEADER( "score_pendingScore", "(Landroid/graphics/Rect;IIII)V" );
|
|
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
jrect, score, playerNum, curTurn, flags );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
}
|
|
|
|
static void
|
|
and_draw_objFinished( DrawCtx* dctx, 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, JCACHE_RECT0, rect );
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
(jint)typ, jrect );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
and_draw_dictChanged( DrawCtx* dctx, XP_S16 playerNum,
|
|
const DictionaryCtxt* dict )
|
|
{
|
|
AndDraw* draw = (AndDraw*)dctx;
|
|
if ( !!dict && !!draw->jdraw ) {
|
|
XP_LOGF( "%s(dict=%p); code=%x", __func__, dict, andDictID(dict) );
|
|
XP_LangCode code = 0; /* A null dict means no-lang */
|
|
if ( NULL != dict ) {
|
|
code = dict_getLangCode( dict );
|
|
}
|
|
/* Don't bother sending repeats. */
|
|
if ( code != draw->curLang ) {
|
|
draw->curLang = code;
|
|
|
|
DRAW_CBK_HEADER( "dictChanged", "(I)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, "<init>", sig ); */
|
|
/* jobject jdict = (*env)->NewObject( env, rclass, initId, (int)dict ); */
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid, (jint)dict );
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef XWFEATURE_MINIWIN
|
|
static const XP_UCHAR*
|
|
and_draw_getMiniWText( DrawCtx* dctx, 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 );
|
|
return draw->miniTextBuf;
|
|
}
|
|
|
|
static void
|
|
and_draw_measureMiniWText( DrawCtx* dctx, const XP_UCHAR* textP,
|
|
XP_U16* width, XP_U16* height )
|
|
{
|
|
DRAW_CBK_HEADER( "measureMiniWText", "(Ljava/lang/String;[I[I)V" );
|
|
|
|
jintArray widthArray = makeIntArray( env, 1, NULL );
|
|
jintArray heightArray = makeIntArray( env, 1, NULL );
|
|
jstring jstr = (*env)->NewStringUTF( env, textP );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
jstr, widthArray, heightArray );
|
|
|
|
deleteLocalRef( env, jstr );
|
|
*width = getIntFromArray( env, widthArray, true );
|
|
*height = getIntFromArray( env, heightArray, true );
|
|
}
|
|
|
|
static void
|
|
and_draw_drawMiniWindow( DrawCtx* dctx, 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, JCACHE_RECT0, rect );
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid,
|
|
jstr, jrect );
|
|
returnJRect( draw, JCACHE_RECT0, jrect );
|
|
deleteLocalRef( env, jstr );
|
|
}
|
|
#endif
|
|
|
|
static XP_Bool
|
|
draw_doNothing( DrawCtx* dctx, ... )
|
|
{
|
|
LOG_FUNC();
|
|
return XP_FALSE;
|
|
} /* draw_doNothing */
|
|
|
|
DrawCtx*
|
|
makeDraw( MPFORMAL EnvThreadInfo* ti, jobject jdraw )
|
|
{
|
|
AndDraw* draw = (AndDraw*)XP_CALLOC( mpool, sizeof(*draw) );
|
|
JNIEnv* env = ENVFORME( ti );
|
|
draw->vtable = XP_MALLOC( mpool, sizeof(*draw->vtable) );
|
|
if ( NULL != jdraw ) {
|
|
draw->jdraw = (*env)->NewGlobalRef( env, jdraw );
|
|
}
|
|
draw->ti = ti;
|
|
MPASSIGN( draw->mpool, mpool );
|
|
|
|
for ( int ii = 0; ii < sizeof(*draw->vtable)/4; ++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;
|
|
}
|
|
|
|
void
|
|
destroyDraw( DrawCtx** dctx )
|
|
{
|
|
if ( !!*dctx ) {
|
|
AndDraw* draw = (AndDraw*)*dctx;
|
|
JNIEnv* env = ENVFORME( draw->ti );
|
|
if ( NULL != draw->jdraw ) {
|
|
(*env)->DeleteGlobalRef( env, draw->jdraw );
|
|
}
|
|
|
|
for ( int ii = 0; ii < JCACHE_COUNT; ++ii ) {
|
|
jobject jobj = draw->jCache[ii];
|
|
if ( !!jobj ) {
|
|
(*env)->DeleteGlobalRef( env, jobj );
|
|
}
|
|
}
|
|
|
|
XP_FREE( draw->mpool, draw->vtable );
|
|
XP_FREE( draw->mpool, draw );
|
|
*dctx = NULL;
|
|
}
|
|
}
|