/* -*-mode: C; fill-column: 77; c-basic-offset: 4; -*- */ /* * Copyright 1999 - 2001 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 #include #include "draw.h" #include "palmmain.h" #include "xwords4defines.h" #include "LocalizedStrIncludes.h" #define CHARRECT_WIDTH 12 #define CHARRECT_HEIGHT 14 #define FONT_HEIGHT 8 #define LINE_SPACING 1 #define SCORE_SEP '.' #define SCORE_SEPSTR "." #define TILE_SUBSCRIPT 1 /* draw tile with numbers below letters? */ static XP_Bool palm_common_draw_drawCell( DrawCtx* p_dctx, const XP_Rect* rect, const XP_UCHAR* letters, XP_Bitmap bitmap, Tile tile, XP_S16 owner, XWBonusType bonus, HintAtts hintAtts, XP_Bool isBlank, XP_Bool isPending, XP_Bool isStar ); static void palm_bnw_draw_score_drawPlayer( DrawCtx* p_dctx, const XP_Rect* rInner, const XP_Rect* rOuter, const DrawScoreInfo* dsi ); static XP_Bool palm_bnw_draw_trayBegin( DrawCtx* p_dctx, const XP_Rect* rect, XP_U16 owner, XP_Bool hasfocus ); static void palm_bnw_draw_trayFinished( DrawCtx* p_dctx ); static void palm_clr_draw_clearRect( DrawCtx* p_dctx, const XP_Rect* rectP ); static void palm_draw_drawMiniWindow( DrawCtx* p_dctx, const XP_UCHAR* text, const XP_Rect* rect, void** closureP ); #define HIGHRES_PUSH_LOC( dctx ) \ { \ XP_U16 oldVal = 0; \ if ( (dctx)->doHiRes ) { \ oldVal = WinSetCoordinateSystem( kCoordinatesNative ); \ } #define HIGHRES_POP_LOC(dctx) \ if ( (dctx)->doHiRes ) { \ (void)WinSetCoordinateSystem( oldVal ); \ (dctx)->oldCoord = 0; \ } \ } #define HIGHRES_PUSH_NOPOP( dctx ) \ if ( (dctx)->doHiRes ) { \ WinSetCoordinateSystem( kCoordinatesNative ); \ } #define HIGHRES_PUSH( dctx ) \ if ( (dctx)->doHiRes ) { \ XP_ASSERT( (dctx)->oldCoord == 0 ); \ (dctx)->oldCoord = WinSetCoordinateSystem( kCoordinatesNative ); \ } #define HIGHRES_POP(dctx) \ if ( (dctx)->doHiRes ) { \ (void)WinSetCoordinateSystem( (dctx)->oldCoord ); \ (dctx)->oldCoord = 0; \ } static void eraseRect( /* PalmDrawCtx* dctx, */const XP_Rect* rect ) { WinEraseRectangle( (const RectangleType*)rect, 0 ); } /* eraseRect */ static void insetRect( XP_Rect* rect, XP_S16 by ) { rect->left += by; rect->top += by; by *= 2; rect->width -= by; rect->height -= by; } /* insetRect */ static void drawBitmapAt( DrawCtx* p_dctx, Int16 resID, Int16 x, Int16 y ) { MemHandle handle; handle = DmGetResource( bitmapRsc, resID ); XP_ASSERT( handle != NULL ); if ( handle != NULL ) { WinDrawBitmap( (BitmapPtr)MemHandleLock(handle), x, y ); XP_ASSERT( MemHandleLockCount(handle ) == 1 ); MemHandleUnlock( handle ); DmReleaseResource( handle ); } } /* drawBitmapAt */ static void bitmapInRect( PalmDrawCtx* dctx, Int16 resID, const XP_Rect* rectP ) { XP_U16 left = rectP->left; XP_U16 top = rectP->top; if ( dctx->globals->gState.showGrid ) { ++left; ++top; } drawBitmapAt( (DrawCtx*)dctx, resID, left, top ); } /* bitmapInRect */ # define BMP_WIDTH 16 # define BMP_HT 16 static void measureFace( PalmDrawCtx* dctx, XP_UCHAR face, PalmFontHtInfo* fhi ) { WinHandle win; BitmapType* bitmap; Coord x, y; Err err; XP_Bool gotIt; XP_U16 top = 0; XP_U16 bottom = 0; XP_UCHAR ch = (XP_UCHAR)face; bitmap = BmpCreate( BMP_WIDTH, BMP_HT, 1, NULL, &err ); if ( err == errNone ) { win = WinCreateBitmapWindow( bitmap, &err ); if ( err == errNone ) { WinHandle oldWin = WinSetDrawWindow( win ); WinSetBackColor( 0 ); /* white */ (void)WinSetTextColor( 1 ); /* black */ (void)WinSetDrawMode( winOverlay ); WinDrawChars( &ch, 1, 0, 0 ); /* Scan from top for top, then from bottom for botton */ gotIt = XP_FALSE; for ( y = 0; !gotIt && y < BMP_HT; ++y ) { for ( x = 0; !gotIt && x < BMP_WIDTH; ++x ) { IndexedColorType pxl = WinGetPixel( x, y ); if ( pxl != 0 ) { top = y; gotIt = XP_TRUE; } } } gotIt = XP_FALSE; for ( y = BMP_HT - 1; !gotIt && y >= 0; --y ) { for ( x = 0; !gotIt && x < BMP_WIDTH; ++x ) { IndexedColorType pxl = WinGetPixel( x, y ); if ( pxl != 0 ) { bottom = y; gotIt = XP_TRUE; } } } (void)WinSetDrawWindow( oldWin ); WinDeleteWindow( win, false ); /* There should be a way to avoid this, but HIGHRES_PUSH after WinSetDrawWindow isn't working... Fix this... */ if ( dctx->doHiRes ) { top *= 2; bottom *= 2; } fhi->topOffset = top; fhi->height = bottom - top + 1; } BmpDelete( bitmap ); } } /* measureFace */ static void checkFontOffsets( PalmDrawCtx* dctx, const DictionaryCtxt* dict ) { XP_LangCode code; XP_U16 nFaces; Tile tile; code = dict_getLangCode( dict ); if ( code != dctx->fontLangCode ) { if ( !!dctx->fontHtInfo ) { XP_FREE( dctx->mpool, dctx->fontHtInfo ); } nFaces = dict_numTileFaces( dict ); dctx->fontHtInfo = XP_MALLOC( dctx->mpool, nFaces * sizeof(*dctx->fontHtInfo) ); for ( tile = 0; tile < nFaces; ++tile ) { XP_UCHAR face[2]; if ( 1 == dict_tilesToString( dict, &tile, 1, face, sizeof(face) ) ) { measureFace( dctx, face[0], &dctx->fontHtInfo[tile] ); } } dctx->fontLangCode = code; } } static XP_Bool palm_common_draw_boardBegin( DrawCtx* p_dctx, const DictionaryCtxt* dict, const XP_Rect* rect, XP_Bool hasfocus ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; PalmAppGlobals* globals = dctx->globals; if ( !globals->gState.showGrid ) { WinDrawRectangleFrame(rectangleFrame, (RectangleType*)rect); } checkFontOffsets( dctx, dict ); return XP_TRUE; } /* palm_common_draw_boardBegin */ #ifdef COLOR_SUPPORT static XP_Bool palm_clr_draw_boardBegin( DrawCtx* p_dctx, const DictionaryCtxt* dict, const XP_Rect* rect, XP_Bool hasfocus ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; WinPushDrawState(); WinSetForeColor( dctx->drawingPrefs->drawColors[COLOR_BLACK] ); WinSetTextColor( dctx->drawingPrefs->drawColors[COLOR_BLACK] ); WinSetBackColor( dctx->drawingPrefs->drawColors[COLOR_WHITE] ); HIGHRES_PUSH_NOPOP(dctx); palm_common_draw_boardBegin( p_dctx, dict, rect, hasfocus ); return XP_TRUE; } /* palm_clr_draw_boardBegin */ static void palm_clr_draw_boardFinished( DrawCtx* p_dctx ) { WinPopDrawState(); } /* palm_clr_draw_boardFinished */ static XP_Bool palm_clr_draw_drawCell( DrawCtx* p_dctx, const XP_Rect* rect, const XP_UCHAR* letters, XP_Bitmap bitmap, Tile tile, XP_S16 owner, XWBonusType bonus, HintAtts hintAtts, XP_Bool isBlank, XP_Bool isPending, XP_Bool isStar ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; IndexedColorType color; XP_U16 index; if ( isPending ) { /* don't color background if will invert */ index = COLOR_WHITE; } else if ( !!bitmap || (!!letters && XP_STRLEN((const char*)letters) > 0)){ index = COLOR_TILE; } else if ( bonus == BONUS_NONE ) { index = COLOR_EMPTY; } else { index = COLOR_DBL_LTTR + bonus - 1; } color = dctx->drawingPrefs->drawColors[index]; WinSetBackColor( color ); if ( !!letters ) { if ( (owner >= 0) && !isPending ) { index = COLOR_PLAYER1 + owner; } else { index = COLOR_BLACK; } color = dctx->drawingPrefs->drawColors[index]; WinSetTextColor( color ); } return palm_common_draw_drawCell( p_dctx, rect, letters, bitmap, tile, owner, bonus, hintAtts, isBlank, isPending, isStar ); } /* palm_clr_draw_drawCell */ static void palm_clr_draw_score_drawPlayer( DrawCtx* p_dctx, const XP_Rect* rInner, const XP_Rect* rOuter, const DrawScoreInfo* dsi ) { XP_U16 playerNum = dsi->playerNum; PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; IndexedColorType newColor; IndexedColorType oldTextColor; IndexedColorType oldFGColor; newColor = dctx->drawingPrefs->drawColors[COLOR_PLAYER1+playerNum]; oldTextColor = WinSetTextColor( newColor ); oldFGColor = WinSetForeColor( newColor ); palm_bnw_draw_score_drawPlayer( p_dctx, rInner, rOuter, dsi ); WinSetTextColor( oldTextColor ); WinSetForeColor( oldFGColor ); } /* palm_clr_draw_score_drawPlayer */ #endif static void palmDrawHintBorders( const XP_Rect* rect, HintAtts hintAtts ) { if ( hintAtts != HINT_BORDER_NONE && hintAtts != HINT_BORDER_CENTER ) { XP_Rect frame = *rect; insetRect( &frame, 1 ); if ( (hintAtts & HINT_BORDER_LEFT) != 0 ) { WinDrawLine( frame.left, frame.top, frame.left, frame.top + frame.height ); } if ( (hintAtts & HINT_BORDER_TOP) != 0 ) { WinDrawLine( frame.left, frame.top, frame.left + frame.width, frame.top ); } if ( (hintAtts & HINT_BORDER_RIGHT) != 0 ) { WinDrawLine( frame.left + frame.width, frame.top, frame.left + frame.width, frame.top + frame.height ); } if ( (hintAtts & HINT_BORDER_BOTTOM) != 0 ) { WinDrawLine( frame.left, frame.top + frame.height, frame.left + frame.width, frame.top + frame.height ); } } } /* palmDrawHintBorders */ static XP_Bool palm_common_draw_drawCell( DrawCtx* p_dctx, const XP_Rect* rect, const XP_UCHAR* letters, XP_Bitmap bitmap, Tile tile, XP_S16 owner, XWBonusType bonus, HintAtts hintAtts, XP_Bool isBlank, XP_Bool isPending, XP_Bool isStar ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; GraphicsAbility able = dctx->able; XP_Rect localR = *rect; XP_U16 len; RectangleType saveClip, intersectR; PalmAppGlobals* globals = dctx->globals; Boolean showGrid = globals->gState.showGrid; Boolean showBonus = bonus != BONUS_NONE; Boolean complete; XP_Bool empty = XP_TRUE; if ( showGrid ) { ++localR.width; ++localR.height; } WinGetClip( &saveClip ); RctGetIntersection( &saveClip, (RectangleType*)&localR, &intersectR ); /* If there's no rect left inside the clip rgn, exit. But if the rect's only partial go ahead and draw, but still return false indicating that we'd like to be allowed to draw again. This is necessary when a cell needs to be redrawn for two reasons, e.g. because its bottom half overlaps a form that's gone away (and that supplied the clip region) and because its top is covered by the trading miniwindow that's also going away. Under no circumstances draw outside the clip region or risk overdrawing menus, other forms, etc. */ if ( intersectR.extent.x == 0 || intersectR.extent.y == 0 ) { return XP_FALSE; } else if ( intersectR.extent.x < localR.width || intersectR.extent.y < localR.height ) { complete = XP_FALSE; } else { complete = XP_TRUE; } WinSetClip( (RectangleType*)&intersectR ); if ( showGrid ) { insetRect( &localR, 1 ); } eraseRect( &localR ); if ( !!letters ) { len = XP_STRLEN( (const char*)letters ); if ( len > 0 ) { XP_S16 strWidth = FntCharsWidth( (const char*)letters, len ); XP_U16 x, y; x = localR.left + ((localR.width-strWidth) / 2); y = localR.top - dctx->fontHtInfo[tile].topOffset; /* '+ 1' below causes us to round up. Without this "Q" is drawn with its top higher than other ASCII letters because it's taller; looks bad. */ y += (localR.height + 1 - dctx->fontHtInfo[tile].height) / 2; if ( len == 1 ) { ++x; } WinDrawChars( (const char*)letters, len, x, y ); showBonus = XP_FALSE; empty = XP_FALSE; } } else if ( !!bitmap ) { XP_Bool doColor = (able == COLOR) && (owner >= 0); XP_U16 x = localR.left+1; XP_U16 y = localR.top+1; /* cheating again; this belongs in a palm_clr method. But the special bitmaps are rare enough that we shouldn't change the palm draw state every time. */ if ( doColor ) { WinSetForeColor( dctx->drawingPrefs->drawColors[COLOR_PLAYER1+owner] ); } if ( dctx->doHiRes ) { ++x; ++y; } WinDrawBitmap( (BitmapPtr)bitmap, x, y ); if ( doColor ) { WinSetForeColor( dctx->drawingPrefs->drawColors[COLOR_BLACK] ); } showBonus = doColor; /* skip bonus in B&W case; can't draw both! */ empty = XP_FALSE; } if ( isStar ) { bitmapInRect( dctx, STAR_BMP_RES_ID, rect ); } else if ( showBonus && (able == ONEBIT) ) { /* this is my one refusal to totally factor bandw and color code */ WinSetPattern( (const CustomPatternType*) &dctx->u.bnw.valuePatterns[bonus-1] ); WinFillRectangle( (RectangleType*)&localR, 0 ); } else if ( !showBonus && empty && !showGrid ) { /* should this be in the v-table so I don't have to test each time? */ RectangleType r; r.topLeft.x = localR.left + ((PALM_BOARD_SCALE-1)/2); r.topLeft.y = localR.top + ((PALM_BOARD_SCALE-1)/2); if ( dctx->doHiRes ) { r.topLeft.x += PALM_BOARD_SCALE/2; r.topLeft.y += PALM_BOARD_SCALE/2; } if ( globals->romVersion >= 35 ) { WinDrawPixel( r.topLeft.x, r.topLeft.y ); } else { r.extent.x = r.extent.y = 1; WinDrawRectangle( &r, 0 ); } } if ( isPending ) { XP_ASSERT( !!bitmap || (!!letters && XP_STRLEN((const char*)letters)>0)); WinInvertRectangle( (RectangleType*)&localR, 0 ); } if ( showGrid ) { WinDrawRectangleFrame(rectangleFrame, (RectangleType*)&localR); } if ( isBlank ) { WinEraseRectangleFrame( roundFrame, (RectangleType*)&localR ); } palmDrawHintBorders( rect, hintAtts ); WinSetClip( &saveClip ); return complete; } /* palm_common_draw_drawCell */ static void palm_draw_invertCell( DrawCtx* p_dctx, const XP_Rect* rect ) { XP_Rect localR = *rect; /* insetRect( &localR, 3 ); */ localR.top += 3; localR.left += 3; localR.width -= 5; localR.height -= 5; HIGHRES_PUSH_LOC( (PalmDrawCtx*)p_dctx ); WinInvertRectangle( (RectangleType*)&localR, 0 ); HIGHRES_POP_LOC( (PalmDrawCtx*)p_dctx ); } /* palm_draw_invertCell */ static XP_Bool palm_clr_draw_trayBegin( DrawCtx* p_dctx, const XP_Rect* rect, XP_U16 owner, XP_Bool hasfocus ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; dctx->trayOwner = owner; WinPushDrawState(); WinSetBackColor( dctx->drawingPrefs->drawColors[COLOR_TILE] ); WinSetTextColor( dctx->drawingPrefs->drawColors[COLOR_PLAYER1+owner] ); WinSetForeColor( dctx->drawingPrefs->drawColors[COLOR_PLAYER1+owner] ); HIGHRES_PUSH_NOPOP(dctx); palm_bnw_draw_trayBegin( p_dctx, rect, owner, hasfocus ); return XP_TRUE; } /* palm_clr_draw_trayBegin */ static XP_Bool palm_bnw_draw_trayBegin( DrawCtx* p_dctx, const XP_Rect* rect, XP_U16 owner, XP_Bool hasfocus ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; WinGetClip( &dctx->oldTrayClip ); WinSetClip( (RectangleType*)rect ); return XP_TRUE; } /* palm_draw_trayBegin */ static void palm_clr_draw_trayFinished( DrawCtx* p_dctx ) { palm_bnw_draw_trayFinished( p_dctx ); WinPopDrawState(); } /* palm_clr_draw_trayFinished */ static void palm_bnw_draw_trayFinished( DrawCtx* p_dctx ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; WinSetClip( &dctx->oldTrayClip ); } /* palm_draw_trayFinished */ static void smallBoldStringAt( const char* str, XP_U16 len, XP_S16 x, XP_U16 y ) { UInt32 oldMode = WinSetScalingMode( kTextScalingOff ); FontID curFont = FntGetFont(); FntSetFont( boldFont ); /* negative x means position backwards from it */ if ( x < 0 ) { x = (-x) - FntCharsWidth( str, len ); } WinDrawChars( str, len, x, y ); FntSetFont( curFont ); WinSetScalingMode( oldMode ); } /* smallBoldStringAt */ static void palm_draw_drawTile( DrawCtx* p_dctx, const XP_Rect* rect, const XP_UCHAR* letters, XP_Bitmap bitmap, XP_S16 val, XP_Bool highlighted ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; char valBuf[3]; XP_Rect localR = *rect; XP_U16 len, width; XP_U16 doubler = 1; if ( dctx->doHiRes ) { doubler = 2; } draw_clearRect( p_dctx, &localR ); localR.width -= 3 * doubler; localR.height -= 3 * doubler; localR.top += 2 * doubler; localR.left += 2 * doubler; /* this will fill it with the tile background color */ WinEraseRectangle( (const RectangleType*)&localR, 0 ); /* Draw the number before the letter. Some PalmOS version don't honor the winOverlay flag and erase. Better to erase the value than the letter. */ if ( val >= 0 ) { (void)StrPrintF( valBuf, "%d", val ); len = XP_STRLEN((const char*)valBuf); if ( 0 ) { } else if ( dctx->doHiRes && dctx->oneDotFiveAvail ) { smallBoldStringAt( valBuf, len, -(localR.left + localR.width), localR.top + localR.height - dctx->fntHeight - 1 ); } else { width = FntCharsWidth( valBuf, len ); WinDrawChars( valBuf, len, localR.left + localR.width - width, localR.top + localR.height - (10*doubler) ); } } if ( !!letters ) { if ( *letters != LETTER_NONE ) { /* blank */ FontID curFont = FntSetFont( largeFont ); HIGHRES_PUSH_LOC( dctx ); WinDrawChars( (char*)letters, 1, localR.left + (1*doubler), localR.top + (0*doubler) ); HIGHRES_POP_LOC(dctx); FntSetFont( curFont ); } } else if ( !!bitmap ) { WinDrawBitmap( (BitmapPtr)bitmap, localR.left+(2*doubler), localR.top+(2*doubler) ); } WinDrawRectangleFrame( rectangleFrame, (RectangleType*)&localR ); if ( highlighted ) { insetRect( &localR, 1 ); WinDrawRectangleFrame(rectangleFrame, (RectangleType*)&localR ); } } /* palm_draw_drawTile */ static void palm_draw_drawTileBack( DrawCtx* p_dctx, const XP_Rect* rect ) { palm_draw_drawTile( p_dctx, rect, (unsigned char*)"?", (XP_Bitmap)NULL, -1, XP_FALSE ); } /* palm_draw_drawTileBack */ static void palm_draw_drawTrayDivider( DrawCtx* p_dctx, const XP_Rect* rect, XP_Bool selected ) { XP_Rect lRect = *rect; draw_clearRect( p_dctx, &lRect ); ++lRect.left; --lRect.width; if ( selected ) { short pattern[] = { 0xFF00, 0xFF00, 0xFF00, 0xFF00 }; WinSetPattern( (const CustomPatternType*)&pattern ); WinFillRectangle( (RectangleType*)&lRect, 0 ); } else { WinDrawRectangle( (RectangleType*)&lRect, 0 ); } } /* palm_draw_drawTrayDivider */ static void palm_bnw_draw_clearRect( DrawCtx* p_dctx, const XP_Rect* rectP ) { eraseRect( rectP ); } /* palm_draw_clearRect */ static void palm_clr_draw_clearRect( DrawCtx* p_dctx, const XP_Rect* rectP ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; IndexedColorType oldColor; oldColor = WinSetBackColor( dctx->drawingPrefs->drawColors[COLOR_WHITE] ); eraseRect( rectP ); WinSetBackColor( oldColor ); } /* palm_clr_draw_clearRect */ static void palm_clr_draw_drawMiniWindow( DrawCtx* p_dctx, const XP_UCHAR* text, const XP_Rect* rect, void** closureP ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; WinSetBackColor( dctx->drawingPrefs->drawColors[COLOR_WHITE] ); WinSetTextColor( dctx->drawingPrefs->drawColors[COLOR_BLACK] ); palm_draw_drawMiniWindow( p_dctx, text, rect, closureP ); } /* palm_clr_draw_drawMiniWindow */ static void palm_draw_drawBoardArrow( DrawCtx* p_dctx, const XP_Rect* rectP, XWBonusType cursorBonus, XP_Bool vertical, HintAtts hintAtts ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; RectangleType oldClip; Int16 resID = vertical? DOWN_ARROW_RESID:RIGHT_ARROW_RESID; WinGetClip( &oldClip ); WinSetClip( (RectangleType*)rectP ); bitmapInRect( dctx, resID, rectP ); palmDrawHintBorders( rectP, hintAtts ); WinSetClip( &oldClip ); } /* palm_draw_drawBoardArrow */ #ifdef COLOR_SUPPORT static void palm_clr_draw_drawBoardArrow( DrawCtx* p_dctx, const XP_Rect* rectP, XWBonusType cursorBonus, XP_Bool vertical, HintAtts hintAtts ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; XP_U16 index; if ( cursorBonus == BONUS_NONE ) { index = COLOR_EMPTY; } else { index = COLOR_DBL_LTTR + cursorBonus - 1; } WinSetBackColor( dctx->drawingPrefs->drawColors[index] ); palm_draw_drawBoardArrow( p_dctx, rectP, cursorBonus, vertical, hintAtts ); } /* palm_clr_draw_drawBoardArrow */ #endif static void palm_draw_scoreBegin( DrawCtx* p_dctx, const XP_Rect* rect, XP_U16 numPlayers, XP_Bool hasfocus ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; HIGHRES_PUSH( dctx ); WinGetClip( &dctx->oldScoreClip ); WinSetClip( (RectangleType*)rect ); eraseRect( rect ); } /* palm_draw_scoreBegin */ /* rectContainsRect: Dup of something in board.c. They could share if I were * willing to link from here out. */ static XP_Bool rectContainsRect( const XP_Rect* rect1, const XP_Rect* rect2 ) { return ( rect1->top <= rect2->top && rect1->left <= rect2->left && rect1->top + rect1->height >= rect2->top + rect2->height && rect1->left + rect1->width >= rect2->left + rect2->width ); } /* rectContainsRect */ static XP_Bool palm_draw_vertScrollBoard( DrawCtx* p_dctx, XP_Rect* rect, XP_S16 dist ) { RectangleType clip; XP_Bool canDoIt; /* if the clip rect doesn't contain the scroll rect we can't do anything right now: WinScrollRectangle won't do its job. */ WinGetClip( &clip ); canDoIt = rectContainsRect( (XP_Rect*)&clip, rect ); if ( canDoIt ) { RectangleType vacated; WinDirectionType dir; if ( dist >= 0 ) { dir = winUp; } else { dir = winDown; dist = -dist; } WinScrollRectangle( (RectangleType*)rect, dir, dist, &vacated ); *rect = *(XP_Rect*)&vacated; } return canDoIt; } /* palm_draw_vertScrollBoard */ /* Given some text, determine its bounds and draw it if requested, else * return the bounds. If the width of the string exceeds that of the rect in * which it can be fit, split it at ':'. */ static void palmMeasureDrawText( PalmDrawCtx* dctx, XP_Rect* bounds, XP_UCHAR* txt, XP_Bool vertical, XP_Bool isTurn, XP_UCHAR skipChar, XP_Bool draw ) { XP_U16 len = XP_STRLEN( (const char*)txt ); XP_U16 widths[2]; XP_U16 maxWidth, height; XP_U16 nLines = 1; XP_U16 secondLen = 0; XP_UCHAR* second = NULL; XP_U16 doubler = 1; if ( dctx->doHiRes ) { doubler = 2; } widths[0] = FntCharsWidth( (const char*)txt, len ) + 1; if ( widths[0] > bounds->width ) { XP_UCHAR ch[2]; ch[0] = skipChar; ch[1] = '\0'; XP_ASSERT( skipChar ); second = (XP_UCHAR*)StrStr( (const char*)txt, (const char*)ch ); XP_ASSERT( !!second ); ++second; /* colon's on the first line */ secondLen = XP_STRLEN( (const char*)second ); len -= secondLen; if ( skipChar ) { --len; } widths[0] = FntCharsWidth( (const char*)txt, len ); widths[1] = FntCharsWidth( (const char*)second, secondLen ); maxWidth = XP_MAX( widths[0], widths[1] ); ++nLines; } else { maxWidth = widths[0]; } height = nLines * FONT_HEIGHT + ( LINE_SPACING * (nLines-1) ); if ( vertical && isTurn ) { height += 5; /* for the horizontal bars */ } height *= doubler; XP_ASSERT( height <= bounds->height ); XP_ASSERT( maxWidth <= bounds->width ); if ( draw ) { XP_U16 x, y; /* Center what we'll be drawing by advancing the appropriate coordinate to eat up half the extra space */ x = bounds->left + 1;// + (bounds->width - widths[0]) / 2; y = bounds->top; if ( vertical && isTurn ) { y += 1; } else { y -= 2; if ( dctx->doHiRes ) { --y; /* tweak it up one high-res pixel */ } } WinDrawChars( (const char*)txt, len, x, y ); if ( nLines == 2 ) { XP_ASSERT( vertical ); y += (FONT_HEIGHT + LINE_SPACING) * doubler; x = bounds->left + ((bounds->width - widths[1]) / 2); WinDrawChars( (const char*)second, secondLen, x, y ); } } else { /* return the measurements */ bounds->width = maxWidth; bounds->height = height; } } /* palmMeasureDrawText */ static void palmFormatRemText( PalmDrawCtx* dctx, XP_UCHAR* buf, XP_S16 nTilesLeft ) { XP_UCHAR* remStr = (*dctx->getResStrFunc)(dctx->globals, STR_REMTILES); if ( nTilesLeft < 0 ) { nTilesLeft = 0; } (void)StrPrintF( (char*)buf, (const char*)remStr, nTilesLeft ); } /* palmFormatRemText */ static void palm_draw_measureRemText( DrawCtx* p_dctx, const XP_Rect* rect, XP_S16 nTilesLeft, XP_U16* widthP, XP_U16* heightP ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; PalmAppGlobals* globals = dctx->globals; XP_UCHAR buf[10]; XP_Rect localRect; XP_Bool isVertical = !globals->gState.showGrid; palmFormatRemText( dctx, buf, nTilesLeft ); localRect = *rect; palmMeasureDrawText( dctx, &localRect, buf, isVertical, XP_FALSE, ':', XP_FALSE ); *widthP = localRect.width; *heightP = localRect.height; } /* palm_draw_measureRemText */ static void palm_draw_drawRemText( DrawCtx* p_dctx, const XP_Rect* rInner, const XP_Rect* rOuter, XP_S16 nTilesLeft ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; PalmAppGlobals* globals = dctx->globals; XP_UCHAR buf[10]; XP_Bool isVertical = !globals->gState.showGrid; palmFormatRemText( dctx, buf, nTilesLeft ); palmMeasureDrawText( dctx, (XP_Rect*)rInner, buf, isVertical, XP_FALSE, ':', XP_TRUE ); } /* palm_draw_drawRemText */ /* Measure text that'll be drawn for player. If vertical, it'll often get * split into two lines, esp after the number of remaining tiles appears. */ static void palmFormatScore( char* buf, const DrawScoreInfo* dsi, XP_Bool vertical ) { char borders[] = {'•', '\0'}; char remBuf[10]; char* remPart = remBuf; if ( vertical || !dsi->isTurn ) { borders[0] = '\0'; } if ( dsi->nTilesLeft >= 0 ) { StrPrintF( remPart, SCORE_SEPSTR "%d", dsi->nTilesLeft ); } else { *remPart = '\0'; } (void)StrPrintF( buf, "%s%d%s%s", borders, dsi->score, remPart, borders ); } /* palmFormatScore */ static void palm_draw_measureScoreText( DrawCtx* p_dctx, const XP_Rect* rect, const DrawScoreInfo* dsi, XP_U16* widthP, XP_U16* heightP ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; PalmAppGlobals* globals = dctx->globals; char buf[20]; /* FontID oldFont = 0; */ XP_Bool vertical = !globals->gState.showGrid; XP_Rect localRect = *rect; /* if ( !vertical && dsi->selected ) { */ /* oldFont = FntGetFont(); */ /* FntSetFont( boldFont ); */ /* } */ palmFormatScore( buf, dsi, vertical ); palmMeasureDrawText( dctx, &localRect, (XP_UCHAR*)buf, dsi->isTurn, vertical, SCORE_SEP, XP_FALSE ); *widthP = localRect.width; *heightP = localRect.height; /* result = widthAndText( buf, score, nTilesInTray, isTurn, */ /* !globals->gState.showGrid, &ignore, ignoreLines ); */ /* if ( !vertical && dsi->selected ) { */ /* FntSetFont( oldFont ); */ /* } */ } /* palm_draw_measureScoreText */ static void palm_bnw_draw_score_drawPlayer( DrawCtx* p_dctx, const XP_Rect* rInner, const XP_Rect* rOuter, const DrawScoreInfo* dsi ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; PalmAppGlobals* globals = dctx->globals; XP_UCHAR scoreBuf[20]; XP_Bool vertical = !globals->gState.showGrid; palmFormatScore( (char*)scoreBuf, dsi, vertical ); palmMeasureDrawText( dctx, (XP_Rect*)rInner, (XP_UCHAR*)scoreBuf, vertical, dsi->isTurn, SCORE_SEP, XP_TRUE ); if ( vertical && dsi->isTurn ) { RectangleType r = *(RectangleType*)rInner; XP_U16 x, y; x = r.topLeft.x; y = r.topLeft.y + 1; WinDrawLine( x, y, x + r.extent.x - 1, y); y += r.extent.y - 3; WinDrawLine( x, y, x + r.extent.x - 1, y ); } if ( dsi->selected ) { WinInvertRectangle( (RectangleType*)rInner, 0 ); /* if ( !vertical ) { */ /* FntSetFont( oldFont ); */ /* } */ } } /* palm_bnw_draw_score_drawPlayer */ #define PENDING_DIGITS 3 static void palm_draw_score_pendingScore( DrawCtx* p_dctx, const XP_Rect* rect, XP_S16 score, XP_U16 playerNum ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; char buf[PENDING_DIGITS+1] = "000"; RectangleType oldClip, newClip; XP_U16 x = rect->left + 1; HIGHRES_PUSH_LOC(dctx); WinGetClip( &oldClip ); RctGetIntersection( &oldClip, (RectangleType*)rect, &newClip ); if ( newClip.extent.y > 0 ) { WinSetClip( &newClip ); eraseRect( rect ); if ( score >= 0 ) { XP_UCHAR tbuf[4]; UInt16 len; if ( score <= 999 ) { StrPrintF( (char*)tbuf, "%d", score ); } else { StrCopy( (char*)tbuf, "wow" ); /* thanks, Marcus :-) */ } len = XP_STRLEN( (const char*)tbuf ); XP_MEMCPY( &buf[PENDING_DIGITS-len], tbuf, len ); } else { StrCopy( buf, "???" ); } /* There's no room for the pts string if we're in highres mode and WinSetScalingMode isn't available. */ if ( !dctx->doHiRes || dctx->oneDotFiveAvail ) { XP_UCHAR* str = (*dctx->getResStrFunc)( dctx->globals, STR_PTS ); if ( dctx->oneDotFiveAvail ) { smallBoldStringAt( (const char*)str, XP_STRLEN((const char*)str), x, rect->top ); } else { WinDrawChars( (const char*)str, XP_STRLEN((const char*)str), x, rect->top ); } } WinDrawChars( buf, PENDING_DIGITS, x, rect->top + (rect->height/2) - 1 ); WinSetClip( &oldClip ); } HIGHRES_POP_LOC(dctx); } /* palm_draw_score_pendingScore */ static void palm_draw_scoreFinished( DrawCtx* p_dctx ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; WinSetClip( &dctx->oldScoreClip ); HIGHRES_POP(dctx); } /* palm_draw_scoreFinished */ static void palmFormatTimerText( XP_UCHAR* buf, XP_S16 secondsLeft ) { XP_U16 minutes, seconds; XP_UCHAR secBuf[6]; if ( secondsLeft < 0 ) { *buf++ = '-'; secondsLeft *= -1; } minutes = secondsLeft / 60; seconds = secondsLeft % 60; /* StrPrintF can't do 0-padding; do it manually. Otherwise 5:03 will come out 5:3 */ StrPrintF( (char*)secBuf, "0%d", seconds ); StrPrintF( (char*)buf, "%d:%s", minutes, secBuf[2] == '\0'? secBuf:&secBuf[1] ); } /* palmFormatTimerText */ static void palm_draw_drawTimer( DrawCtx* p_dctx, const XP_Rect* rInner, const XP_Rect* rOuter, XP_U16 player, XP_S16 secondsLeft ) { /* This is called both from within drawScoreboard and not, meaning that * sometimes draw_boardBegin() and draw_boardFinished() bracket it and * sometimes they don't. So it has to do its own HIGHRES_PUSH_LOC stuff. */ PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; XP_UCHAR buf[10]; XP_Rect localR = *rInner; RectangleType saveClip; XP_U16 len, width, y; HIGHRES_PUSH_LOC(dctx); palmFormatTimerText( buf, secondsLeft ); len = XP_STRLEN( (const char*)buf ); width = FntCharsWidth( (const char*)buf, len ); eraseRect( &localR ); if ( width < localR.width ) { localR.left += localR.width - width; localR.width = width; } y = localR.top - 2; if ( dctx->doHiRes ) { y -= 1; /* tweak it up one high-res pixel */ } WinGetClip( &saveClip ); WinSetClip( (RectangleType*)&localR ); WinDrawChars( (const char*)buf, len, localR.left, y ); WinSetClip( &saveClip ); HIGHRES_POP_LOC(dctx); } /* palm_draw_drawTimer */ #define MINI_LINE_HT 12 #define MINI_V_PADDING 6 #define MINI_H_PADDING 8 static XP_UCHAR* palm_draw_getMiniWText( DrawCtx* p_dctx, XWMiniTextType textHint ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; XP_UCHAR* str; XP_U16 strID = 0; /* make compiler happy */ switch( textHint ) { case BONUS_DOUBLE_LETTER: strID = STR_DOUBLE_LETTER; break; case BONUS_DOUBLE_WORD: strID = STR_DOUBLE_WORD; break; case BONUS_TRIPLE_LETTER: strID = STR_TRIPLE_LETTER; break; case BONUS_TRIPLE_WORD: strID = STR_TRIPLE_WORD; break; case INTRADE_MW_TEXT: strID = STR_TRADING_REMINDER; break; default: XP_ASSERT( XP_FALSE ); } str = (*dctx->getResStrFunc)( dctx->globals, strID ); return str; } /* palm_draw_getMiniWText */ static void splitString( const XP_UCHAR* str, XP_U16* nBufsP, XP_UCHAR** bufs ) { XP_U16 nBufs = 0; for ( ; ; ) { XP_UCHAR* nextStr = StrStr( str, XP_CR ); XP_U16 len = nextStr==NULL? XP_STRLEN(str): nextStr - str; XP_ASSERT( nextStr != str ); XP_MEMCPY( bufs[nBufs], str, len ); bufs[nBufs][len] = '\0'; ++nBufs; if ( nextStr == NULL ) { break; } str = nextStr + XP_STRLEN(XP_CR); /* skip '\n' */ } *nBufsP = nBufs; } /* splitString */ #define VALUE_HINT_RECT_HEIGHT 12 #define VALUE_HINT_RECT_HEIGHT_HR 24 static XP_U16 getMiniLineHt( PalmDrawCtx* dctx ) { if ( dctx->doHiRes ) { return VALUE_HINT_RECT_HEIGHT_HR; } else { return VALUE_HINT_RECT_HEIGHT; } } /* getMiniLineHt */ static void palm_draw_measureMiniWText( DrawCtx* p_dctx, const XP_UCHAR* str, XP_U16* widthP, XP_U16* heightP ) { XP_U16 maxWidth, height; XP_UCHAR buf1[48]; XP_UCHAR buf2[48]; XP_UCHAR* bufs[2] = { buf1, buf2 }; XP_U16 nBufs, i; HIGHRES_PUSH_LOC( (PalmDrawCtx*)p_dctx ); FntSetFont( stdFont ); splitString( str, &nBufs, bufs ); for ( height = 0, maxWidth = 0, i = 0; i < nBufs; ++i ) { XP_U16 oneWidth = 8 + FntCharsWidth( (const char*)bufs[i], XP_STRLEN(bufs[i]) ); maxWidth = XP_MAX( maxWidth, oneWidth ); height += getMiniLineHt((PalmDrawCtx*)p_dctx); } *widthP = maxWidth; *heightP = height + 3; HIGHRES_POP_LOC( (PalmDrawCtx*)p_dctx ); } /* palm_draw_measureMiniWText */ typedef struct PalmMiniWinData { WinHandle bitsBehind; XP_S16 miniX; XP_S16 miniY; } PalmMiniWinData; static void palm_draw_drawMiniWindow( DrawCtx* p_dctx, const XP_UCHAR* text, const XP_Rect* rect, void** closureP ) { RectangleType localR = *(RectangleType*)rect; XP_UCHAR buf1[48]; XP_UCHAR buf2[48]; XP_UCHAR* bufs[2] = { buf1, buf2 }; XP_U16 nBufs, i, offset; XP_U16 ignoreErr; XP_Bool hasClosure = !!closureP; PalmMiniWinData* data = (PalmMiniWinData*)(hasClosure? *closureP: NULL); PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; HIGHRES_PUSH_LOC(dctx); if ( hasClosure ) { if ( !data ) { /* capture a bit extra to avoid ghosting on hires devices */ localR = *(RectangleType*)rect; insetRect( (XP_Rect*)&localR, -2 ); data = XP_MALLOC( dctx->mpool, sizeof(PalmMiniWinData) ); data->bitsBehind = WinSaveBits( &localR, &ignoreErr ); data->miniX = localR.topLeft.x; data->miniY = localR.topLeft.y; *closureP = data; } else { XP_ASSERT( data->miniX == localR.topLeft.x ); XP_ASSERT( data->miniY == localR.topLeft.y ); } } localR = *(RectangleType*)rect; WinEraseRectangle( &localR, 0 ); localR.topLeft.x++; localR.topLeft.y++; localR.extent.x -= 3; localR.extent.y -= 3; WinDrawRectangleFrame( popupFrame, &localR ); splitString( text, &nBufs, bufs ); offset = 0; for ( i = 0; i < nBufs; ++i ) { XP_UCHAR* txt = bufs[i]; XP_U16 len = XP_STRLEN( txt ); XP_U16 width = FntCharsWidth( txt, len ); WinDrawChars( (const char*)txt, len, localR.topLeft.x + ((localR.extent.x-width)/2), localR.topLeft.y + 1 + offset ); offset += getMiniLineHt( dctx ); } HIGHRES_POP_LOC( dctx ); } /* palm_draw_drawMiniWindow */ static void palm_draw_eraseMiniWindow( DrawCtx* p_dctx, const XP_Rect* rect, XP_Bool lastTime, void** closure, XP_Bool* invalUnder ) { PalmMiniWinData* data = (PalmMiniWinData*)*closure; PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; if ( !!closure && !!*closure ) { HIGHRES_PUSH_LOC(dctx); /* this DELETES data->bitsBehind */ WinRestoreBits( data->bitsBehind, data->miniX, data->miniY ); HIGHRES_POP_LOC(dctx); XP_FREE( dctx->mpool, data ); *closure = NULL; } } /* palm_draw_eraseMiniWindow */ static void draw_doNothing( DrawCtx* dctx, ... ) { } /* draw_doNothing */ DrawCtx* palm_drawctxt_make( MPFORMAL GraphicsAbility able, PalmAppGlobals* globals, GetResStringFunc getRSF, DrawingPrefs* drawprefs ) { PalmDrawCtx* dctx; XP_U16 i; XP_U16 cWinWidth, cWinHeight; dctx = XP_MALLOC( mpool, sizeof(PalmDrawCtx) ); XP_MEMSET( dctx, 0, sizeof(PalmDrawCtx) ); MPASSIGN(dctx->mpool, mpool); dctx->able = able; dctx->doHiRes = globals->hasHiRes && globals->width == 320; dctx->globals = globals; dctx->getResStrFunc = getRSF; dctx->drawingPrefs = drawprefs; dctx->vtable = XP_MALLOC( mpool, sizeof(*(((PalmDrawCtx*)dctx)->vtable)) ); for ( i = 0; i < sizeof(*dctx->vtable)/4; ++i ) { ((void**)(dctx->vtable))[i] = draw_doNothing; } /* To keep the number of entry points this file has to 1, all functions but the initter called from here must go through a vtable call. so....*/ dctx->drawBitmapFunc = drawBitmapAt; SET_VTABLE_ENTRY( dctx->vtable, draw_invertCell, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawTile, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawTileBack, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawTrayDivider, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawBoardArrow, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_scoreBegin, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_vertScrollBoard, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_measureRemText, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawRemText, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_measureScoreText, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_score_pendingScore, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_scoreFinished, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawTimer, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_getMiniWText, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_measureMiniWText, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawMiniWindow, palm ); SET_VTABLE_ENTRY( dctx->vtable, draw_eraseMiniWindow, palm ); if ( able == COLOR ) { #ifdef COLOR_SUPPORT SET_VTABLE_ENTRY( dctx->vtable, draw_boardBegin, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_boardFinished, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawCell, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_score_drawPlayer, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_trayBegin, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_trayFinished, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_clearRect, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawMiniWindow, palm_clr ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawBoardArrow, palm_clr ); #else XP_ASSERT(0); #endif } else { SET_VTABLE_ENTRY( dctx->vtable, draw_boardBegin, palm_common ); SET_VTABLE_ENTRY( dctx->vtable, draw_drawCell, palm_common ); SET_VTABLE_ENTRY( dctx->vtable, draw_score_drawPlayer, palm_bnw ); SET_VTABLE_ENTRY( dctx->vtable, draw_trayBegin, palm_bnw ); SET_VTABLE_ENTRY( dctx->vtable, draw_trayFinished, palm_bnw ); SET_VTABLE_ENTRY( dctx->vtable, draw_clearRect, palm_bnw ); } cWinWidth = CHARRECT_WIDTH; cWinHeight = CHARRECT_HEIGHT; if ( dctx->doHiRes ) { cWinWidth *= 2; cWinHeight *= 2; } if ( able == COLOR ) { } else { short patBits[] = { 0x8844, 0x2211, 0x8844, 0x2211, 0xaa55, 0xaa55, 0xaa55, 0xaa55, 0xCC66, 0x3399, 0xCC66, 0x3399, 0xCCCC, 0x3333, 0xCCCC, 0x3333 }; XP_MEMCPY( &dctx->u.bnw.valuePatterns[0], patBits, sizeof(patBits) ); } dctx->fntHeight = FntBaseLine(); dctx->oneDotFiveAvail = globals->oneDotFiveAvail; return (DrawCtx*)dctx; } /* palm_drawctxt_make */ void palm_drawctxt_destroy( DrawCtx* p_dctx ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; XP_FREE( dctx->mpool, p_dctx->vtable ); if ( !!dctx->fontHtInfo ) { XP_FREE( dctx->mpool, dctx->fontHtInfo ); } XP_FREE( dctx->mpool, dctx ); } /* palm_drawctxt_destroy */