mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-10 05:26:10 +01:00
52d2694bc7
Add a third processor type to the .so, and fix first compile-time warnings and then a few dumb bugs based on assumptions about ptr sizes. Works to play networked games and browse wordlists, but is not extensively tested and needs to be before release.
721 lines
24 KiB
C
721 lines
24 KiB
C
/* -*-mode: C; compile-command: "find-and-gradle.sh insXwdDeb"; -*- */
|
|
/*
|
|
* 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, sizeof(jarr[0]) );
|
|
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 = (*env)->NewIntArray( env, 1 );
|
|
jintArray heightArray = (*env)->NewIntArray( env, 1 );
|
|
jobject jrect = makeJRect( draw, JCACHE_RECT0, rect );
|
|
|
|
jboolean 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 );
|
|
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 = (*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;
|
|
} /* 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", "(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, "<init>", sig ); */
|
|
/* jobject jdict = (*env)->NewObject( env, rclass, initId, (int)dict ); */
|
|
|
|
(*env)->CallVoidMethod( env, draw->jdraw, mid, (jlong)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 = (*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;
|
|
}
|
|
|
|
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)/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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|