/* -*-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; JNIEnv** env; jobject jdraw; /* global ref; free it! */ XP_LangCode curLang; jobject jCache[JCACHE_COUNT]; XP_UCHAR miniTextBuf[128]; MPSLOT } AndDraw; static jobject makeJRect( AndDraw* draw, int indx, const XP_Rect* rect ) { JNIEnv* env = *draw->env; jobject robj = draw->jCache[indx]; 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, "", "(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 ); } return robj; } /* makeJRect */ #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[] ) { XP_U16 ii; JNIEnv* env = *draw->env; 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 ( 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[] ) { XP_U16 ii; JNIEnv* env = *draw->env; 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 ( 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 ( 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 = *draw->env; jobject dsiobj = draw->jCache[indx]; if ( !dsiobj ) { jclass rclass = (*env)->FindClass( env, PKG_PATH("jni/DrawScoreInfo") ); jmethodID initId = (*env)->GetMethodID( env, rclass, "", "()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 = *draw->env; \ 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]; int ii; for ( 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 ); 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 ); } return result; } static void and_draw_score_drawPlayers( DrawCtx* dctx, const XP_Rect* scoreRect, XP_U16 nPlayers, DrawScoreInfo playerData[], XP_Rect playerRects[] ) { XP_U16 ii; 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 ( ii = 0; ii < nPlayers; ++ii ) { jobject jrect = (*env)->GetObjectArrayElement( env, jrects, ii ); readJRect( env, &playerRects[ii], 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 ); } 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 ); } 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 ); *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 ); } /* 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 ); } } static XP_Bool and_draw_boardBegin( DrawCtx* dctx, const XP_Rect* rect, XP_U16 cellWidth, XP_U16 cellHeight, DrawFocusState XP_UNUSED(dfs) ) { DRAW_CBK_HEADER( "boardBegin", "(Landroid/graphics/Rect;II)Z" ); jobject jrect = makeJRect( draw, JCACHE_RECT0, rect ); jboolean result = (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, cellWidth, cellHeight ); return result; } 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 ); 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 ); } 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 ); 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 ); 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 ); 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 ); return (*env)->CallBooleanMethod( env, draw->jdraw, mid, jrect, flags ); } 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 ); } 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 ); } 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 ); #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, "", 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 ); deleteLocalRef( env, jstr ); } #endif static XP_Bool draw_doNothing( DrawCtx* dctx, ... ) { LOG_FUNC(); return XP_FALSE; } /* draw_doNothing */ DrawCtx* makeDraw( MPFORMAL JNIEnv** envp, jobject jdraw ) { AndDraw* draw = (AndDraw*)XP_CALLOC( mpool, sizeof(*draw) ); JNIEnv* env = *envp; draw->vtable = XP_MALLOC( mpool, sizeof(*draw->vtable) ); if ( NULL != jdraw ) { draw->jdraw = (*env)->NewGlobalRef( env, jdraw ); } draw->env = envp; MPASSIGN( draw->mpool, mpool ); int ii; for ( 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(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 = *draw->env; if ( NULL != draw->jdraw ) { (*env)->DeleteGlobalRef( env, draw->jdraw ); } int ii; for ( 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; } }