/* -*-mode: C; fill-column: 77; c-basic-offset: 4; -*- */ /* * Copyright 1999 - 2001 by Eric House (fixin@peak.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, XP_Rect* rect, XP_UCHAR* letters, XP_Bitmap bitmap, 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, XP_S16 playerNum, XP_Rect* rInner, XP_Rect* rOuter, DrawScoreInfo* dsi ); static XP_Bool palm_bnw_draw_trayBegin( DrawCtx* p_dctx, 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, XP_Rect* rectP ); static void palm_draw_drawMiniWindow( DrawCtx* p_dctx, unsigned char* text, XP_Rect* rect, void** closureP ); #ifdef FEATURE_HIGHRES #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; \ } #else #define HIGHRES_PUSH(dctx) #define HIGHRES_PUSH_LOC(dctx) #define HIGHRES_PUSH_NOPOP(dctx) #define HIGHRES_POP(dctx) #define HIGHRES_POP_LOC(dctx) #endif static void eraseRect( /* PalmDrawCtx* dctx, */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, 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 */ static XP_Bool palm_common_draw_boardBegin( DrawCtx* p_dctx, XP_Rect* rect, XP_Bool hasfocus ) { PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; PalmAppGlobals* globals = dctx->globals; if ( !globals->gState.showGrid ) { WinDrawRectangleFrame(rectangleFrame, (RectangleType*)rect); } return XP_TRUE; } /* palm_common_draw_boardBegin */ #ifdef COLOR_SUPPORT static XP_Bool palm_clr_draw_boardBegin( DrawCtx* p_dctx, 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, 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, XP_Rect* rect, XP_UCHAR* letters, XP_Bitmap bitmap, 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, owner, bonus, hintAtts, isBlank, isPending, isStar ); } /* palm_clr_draw_drawCell */ static void palm_clr_draw_score_drawPlayer( DrawCtx* p_dctx, XP_S16 playerNum, XP_Rect* rInner, XP_Rect* rOuter, DrawScoreInfo* dsi ) { 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, playerNum, rInner, rOuter, dsi ); WinSetTextColor( oldTextColor ); WinSetForeColor( oldFGColor ); } /* palm_clr_draw_score_drawPlayer */ #endif static void palmDrawHintBorders( 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, XP_Rect* rect, XP_UCHAR* letters, XP_Bitmap bitmap, 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 = localR.left + ((localR.width-strWidth) / 2); XP_U16 y = localR.top-1; if ( len == 1 ) { ++x; } if ( dctx->doHiRes ) { --y; } WinDrawChars( (const char*)letters, len, x, y ); showBonus = XP_FALSE; empty = XP_FALSE; } } else if ( !!bitmap ) { XP_Bool doColor = (able == COLOR) && (owner >= 0); /* 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] ); } WinDrawBitmap( (BitmapPtr)bitmap, localR.left+1, localR.top+1 ); 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); #ifdef FEATURE_HIGHRES if ( dctx->doHiRes ) { r.topLeft.x += PALM_BOARD_SCALE/2; r.topLeft.y += PALM_BOARD_SCALE/2; } #endif 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, 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, 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, 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 */ #ifdef FEATURE_HIGHRES 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 */ #endif static void palm_draw_drawTile( DrawCtx* p_dctx, XP_Rect* rect, 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; #ifdef FEATURE_HIGHRES if ( dctx->doHiRes ) { doubler = 2; } #endif 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 ) { #ifdef FEATURE_HIGHRES } else if ( dctx->doHiRes && dctx->oneDotFiveAvail ) { smallBoldStringAt( valBuf, len, -(localR.left + localR.width), localR.top + localR.height - dctx->fntHeight - 1 ); #endif } 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 */ RectangleType charRect = {{0,0}, {CHARRECT_WIDTH*2, CHARRECT_HEIGHT*2}}; FontID curFont = FntGetFont(); WinHandle curWind; WinHandle offScreenCharWin = dctx->offScreenCharWin; FntSetFont( largeFont ); XP_ASSERT( !!offScreenCharWin ); curWind = WinSetDrawWindow( offScreenCharWin ); HIGHRES_PUSH_LOC( dctx ); WinEraseRectangle( &charRect, 0 ); WinDrawChars( (char*)letters, 1, 0, 0 ); width = FntCharsWidth( letters, 1 ); (void)WinSetDrawWindow( curWind ); HIGHRES_POP_LOC(dctx); charRect.extent.x = width; WinCopyRectangle( offScreenCharWin, 0, &charRect, localR.left + (1*doubler), localR.top + (0*doubler), winOverlay ); 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, 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, 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, XP_Rect* rectP ) { eraseRect( rectP ); } /* palm_draw_clearRect */ static void palm_clr_draw_clearRect( DrawCtx* p_dctx, 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, unsigned char* text, 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, 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, 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, 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( XP_Rect* rect1, 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; #ifdef FEATURE_HIGHRES if ( dctx->doHiRes ) { doubler = 2; } #endif 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; } 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, 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, XP_Rect* rInner, 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, 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, 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, XP_Rect* rect, 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, XP_S16 playerNum, XP_Rect* rInner, XP_Rect* rOuter, 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, 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, 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, XP_Rect* rInner, XP_Rect* rOuter, XP_U16 player, XP_S16 secondsLeft ) { XP_UCHAR buf[10]; XP_Rect localR = *rInner; RectangleType saveClip; XP_U16 len, width; 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; } localR.top -= 2; localR.height += 1; WinGetClip( &saveClip ); WinSetClip( (RectangleType*)&localR ); WinDrawChars( (const char*)buf, len, localR.left, localR.top ); WinSetClip( &saveClip ); } /* 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( 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 ( 0 ) { #ifdef FEATURE_HIGHRES } else if ( dctx->doHiRes ) { return VALUE_HINT_RECT_HEIGHT_HR; #endif } else { return VALUE_HINT_RECT_HEIGHT; } } /* getMiniLineHt */ static void palm_draw_measureMiniWText( DrawCtx* p_dctx, unsigned char* 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, unsigned char* text, 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, XP_Rect* rect, XP_Bool lastTime, void** closure, XP_Bool* invalUnder ) { PalmMiniWinData* data = (PalmMiniWinData*)*closure; #if defined MEM_DEBUG || defined FEATURE_HIGHRES PalmDrawCtx* dctx = (PalmDrawCtx*)p_dctx; #endif 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; Err ignore; dctx = XP_MALLOC( mpool, sizeof(PalmDrawCtx) ); XP_MEMSET( dctx, 0, sizeof(PalmDrawCtx) ); MPASSIGN(dctx->mpool, mpool); dctx->able = able; #ifdef FEATURE_HIGHRES dctx->doHiRes = globals->hasHiRes && globals->width == 320; #endif 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; #ifdef FEATURE_HIGHRES if ( dctx->doHiRes ) { cWinWidth *= 2; cWinHeight *= 2; } #endif dctx->offScreenCharWin = WinCreateOffscreenWindow( cWinWidth, cWinHeight, nativeFormat, &ignore ); 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_ASSERT( !!dctx->offScreenCharWin ); WinDeleteWindow( dctx->offScreenCharWin, false ); XP_FREE( dctx->mpool, p_dctx->vtable ); XP_FREE( dctx->mpool, dctx ); } /* palm_drawctxt_destroy */