mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-07 05:24:46 +01:00
7dc949fb38
Make on demand rather than guessing. Only wind up with six or so, so the linear search is probably ok.
680 lines
22 KiB
C
680 lines
22 KiB
C
/* -*- compile-command: "cd ../linux && make MEMDEBUG=TRUE -j3"; -*- */
|
|
/*
|
|
* Copyright 1997 - 2021 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.
|
|
*/
|
|
|
|
/* Re: boards that can't fit on the screen. Let's have an assumption, that
|
|
* the tray is always either below the board or overlapping its bottom. There
|
|
* is never any board visible below the tray. But it's possible to have a
|
|
* board small enough that scrolling is necessary even with the tray hidden.
|
|
*
|
|
* Currently we don't specify the board bounds. We give top,left and the size
|
|
* of cells, and the board figures out the bounds. That's probably a mistake.
|
|
* Better to give bounds, and maybe a min scale, and let it figure out how
|
|
* many cells can be visible. Could it also decide if the tray should overlap
|
|
* or be below? Some platforms have to own that decision since the tray is
|
|
* narrower than the board. So give them separate bounds-setting functions,
|
|
* and let the board code figure out if they overlap.
|
|
*
|
|
* Problem: the board size must always be a multiple of the scale. The
|
|
* platform-specific code has an easy time doing that math. The board can't:
|
|
* it'd have to take bounds, then spit them back out slightly modified. It'd
|
|
* also have to refuse to work (maybe just assert) if asked to take bounds
|
|
* before it had a min_scale.
|
|
*
|
|
* Another way of looking at it closer to the current: the board's position
|
|
* and the tray's bounds determine the board's bounds. If the board's vScale
|
|
* times the number of rows places its would-be bottom at or above the bottom
|
|
* of the tray, then it's potentially visible. If its would-be bottom is
|
|
* above the top of the tray, no scrolling is needed. But if it's below the
|
|
* tray entirely then scrolling will happen even with the tray hidden. As
|
|
* above, we assume the board never appears below the tray.
|
|
*/
|
|
|
|
#include "comtypes.h"
|
|
#include "board.h"
|
|
#include "scorebdp.h"
|
|
#include "game.h"
|
|
#include "server.h"
|
|
#include "comms.h" /* for CHANNEL_NONE */
|
|
#include "dictnry.h"
|
|
#include "draw.h"
|
|
#include "engine.h"
|
|
#include "util.h"
|
|
#include "mempool.h" /* debug only */
|
|
#include "memstream.h"
|
|
#include "strutils.h"
|
|
#include "LocalizedStrIncludes.h"
|
|
|
|
#include "boardp.h"
|
|
#include "dragdrpp.h"
|
|
#include "dbgutil.h"
|
|
|
|
#ifdef CPLUS
|
|
extern "C" {
|
|
#endif
|
|
|
|
static XP_Bool drawCell( BoardCtxt* board, XWEnv xwe, XP_U16 col, XP_U16 row,
|
|
XP_Bool skipBlanks );
|
|
static void drawBoard( BoardCtxt* board, XWEnv xwe );
|
|
static void scrollIfCan( BoardCtxt* board, XWEnv xwe );
|
|
#ifdef KEYBOARD_NAV
|
|
static XP_Bool cellFocused( const BoardCtxt* board, XP_U16 col, XP_U16 row );
|
|
#endif
|
|
#ifdef XWFEATURE_MINIWIN
|
|
static void drawTradeWindowIf( BoardCtxt* board );
|
|
#else
|
|
# define drawTradeWindowIf( board )
|
|
#endif
|
|
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static HintAtts figureHintAtts( BoardCtxt* board, XP_U16 col, XP_U16 row );
|
|
#else
|
|
# define figureHintAtts(b,c,r) HINT_BORDER_NONE
|
|
#endif
|
|
|
|
// #define LOG_CELL_DRAW
|
|
#ifdef LOG_CELL_DRAW
|
|
static XP_UCHAR* formatFlags(XP_UCHAR* buf, XP_U16 len, CellFlags flags);
|
|
#else
|
|
# define formatFlags(buf, siz, flags) ""
|
|
#endif
|
|
|
|
#ifdef POINTER_SUPPORT
|
|
static void drawDragTileIf( BoardCtxt* board, XWEnv xwe );
|
|
#endif
|
|
|
|
#ifdef KEYBOARD_NAV
|
|
#ifdef PERIMETER_FOCUS
|
|
static void
|
|
invalOldPerimeter( BoardCtxt* board )
|
|
{
|
|
/* We need to inval the center of the row that's moving into the center
|
|
from a border (at which point it got borders drawn on it.) */
|
|
ScrollData* vsd = &board->sd[SCROLL_V];
|
|
XP_S16 diff = vsd->offset - board->prevYScrollOffset;
|
|
XP_U16 firstRow, lastRow;
|
|
XP_ASSERT( diff != 0 );
|
|
if ( diff < 0 ) {
|
|
/* moving up; inval row previously on bottom */
|
|
firstRow = vsd->offset + 1;
|
|
lastRow = board->prevYScrollOffset;
|
|
} else {
|
|
XP_U16 nVisible = vsd->lastVisible - vsd->offset + 1;
|
|
lastRow = board->prevYScrollOffset + nVisible - 1;
|
|
firstRow = lastRow - diff + 1;
|
|
}
|
|
XP_ASSERT( firstRow <= lastRow );
|
|
while ( firstRow <= lastRow ) {
|
|
board->redrawFlags[firstRow] |= ~0;
|
|
++firstRow;
|
|
}
|
|
} /* invalOldPerimeter */
|
|
#endif
|
|
#endif
|
|
|
|
/* if any of a blank's neighbors is invalid, so must the blank become (since
|
|
* they share a border and drawing the neighbor will redraw the blank's border
|
|
* too) We'll want to redraw only those blanks that are themselves already
|
|
* invalid *OR* that become invalid this way, and so we'll build a new
|
|
* BlankQueue of them and replace the old.
|
|
*
|
|
* I'm not sure what happens if two blanks are neighbors.
|
|
*/
|
|
#define INVAL_BIT_SET(b,c,r) (((b)->redrawFlags[(r)] & (1 <<(c))) != 0)
|
|
static void
|
|
invalBlanksWithNeighbors( BoardCtxt* board, BlankQueue* bqp )
|
|
{
|
|
XP_U16 ii;
|
|
XP_U16 lastCol, lastRow;
|
|
BlankQueue invalBlanks;
|
|
XP_U16 nInvalBlanks = 0;
|
|
|
|
lastCol = model_numCols(board->model) - 1;
|
|
lastRow = model_numRows(board->model) - 1;
|
|
|
|
for ( ii = 0; ii < bqp->nBlanks; ++ii ) {
|
|
XP_U16 modelCol = bqp->col[ii];
|
|
XP_U16 modelRow = bqp->row[ii];
|
|
XP_U16 col, row;
|
|
|
|
flipIf( board, modelCol, modelRow, &col, &row );
|
|
|
|
if ( INVAL_BIT_SET( board, col, row )
|
|
|| (col > 0 && INVAL_BIT_SET( board, col-1, row ))
|
|
|| (col < lastCol && INVAL_BIT_SET( board, col+1, row ))
|
|
|| (row > 0 && INVAL_BIT_SET( board, col, row-1 ))
|
|
|| (row < lastRow && INVAL_BIT_SET( board, col, row+1 )) ) {
|
|
|
|
invalCell( board, col, row );
|
|
|
|
invalBlanks.col[nInvalBlanks] = (XP_U8)col;
|
|
invalBlanks.row[nInvalBlanks] = (XP_U8)row;
|
|
++nInvalBlanks;
|
|
}
|
|
}
|
|
invalBlanks.nBlanks = nInvalBlanks;
|
|
XP_MEMCPY( bqp, &invalBlanks, sizeof(*bqp) );
|
|
} /* invalBlanksWithNeighbors */
|
|
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static HintAtts
|
|
figureHintAtts( BoardCtxt* board, XP_U16 col, XP_U16 row )
|
|
{
|
|
HintAtts result = HINT_BORDER_NONE;
|
|
|
|
/* while lets us break to exit... */
|
|
while ( board->trayVisState == TRAY_REVEALED
|
|
&& !board->gi->hintsNotAllowed
|
|
&& board->gi->allowHintRect ) {
|
|
BdHintLimits limits;
|
|
if ( dragDropGetHintLimits( board, &limits ) ) {
|
|
/* do nothing */
|
|
} else if ( board->selInfo->hasHintRect ) {
|
|
limits = board->selInfo->limits;
|
|
} else {
|
|
break;
|
|
}
|
|
|
|
if ( col < limits.left ) break;
|
|
if ( row < limits.top ) break;
|
|
if ( col > limits.right ) break;
|
|
if ( row > limits.bottom ) break;
|
|
|
|
if ( col == limits.left ) {
|
|
result |= HINT_BORDER_LEFT;
|
|
}
|
|
if ( col == limits.right ) {
|
|
result |= HINT_BORDER_RIGHT;
|
|
}
|
|
if ( row == limits.top) {
|
|
result |= HINT_BORDER_TOP;
|
|
}
|
|
if ( row == limits.bottom ) {
|
|
result |= HINT_BORDER_BOTTOM;
|
|
}
|
|
#ifndef XWFEATURE_SEARCHLIMIT_DOCENTERS
|
|
if ( result == HINT_BORDER_NONE ) {
|
|
result = HINT_BORDER_CENTER;
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
} /* figureHintAtts */
|
|
#endif
|
|
|
|
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 */
|
|
|
|
#ifdef XWFEATURE_MINIWIN
|
|
static void
|
|
makeMiniWindowForTrade( BoardCtxt* board )
|
|
{
|
|
const XP_UCHAR* text;
|
|
|
|
text = draw_getMiniWText( board->draw, INTRADE_MW_TEXT );
|
|
|
|
makeMiniWindowForText( board, text, MINIWINDOW_TRADING );
|
|
} /* makeMiniWindowForTrade */
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
static CellFlags
|
|
flagsForCrosshairs( const BoardCtxt* board, XP_U16 col, XP_U16 row )
|
|
{
|
|
CellFlags flags = 0;
|
|
if ( ! board->hideCrosshairs ) {
|
|
XP_Bool inHor, inVert;
|
|
dragDropInCrosshairs( board, col, row, &inHor, &inVert );
|
|
if ( inHor ) {
|
|
flags |= CELL_CROSSHOR;
|
|
}
|
|
if ( inVert ) {
|
|
flags |= CELL_CROSSVERT;
|
|
}
|
|
}
|
|
return flags;
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
drawBoard( BoardCtxt* board, XWEnv xwe )
|
|
{
|
|
if ( board->needsDrawing
|
|
&& draw_boardBegin( board->draw, xwe, &board->boardBounds,
|
|
board->sd[SCROLL_H].scale,
|
|
board->sd[SCROLL_V].scale,
|
|
dfsFor( board, OBJ_BOARD ), board->tvType ) ) {
|
|
|
|
XP_Bool allDrawn = XP_TRUE;
|
|
XP_S16 ii;
|
|
XP_S16 col, row, nVisCols;
|
|
ModelCtxt* model = board->model;
|
|
BoardArrow const* arrow = NULL;
|
|
ScrollData* hsd = &board->sd[SCROLL_H];
|
|
ScrollData* vsd = &board->sd[SCROLL_V];
|
|
BlankQueue bq;
|
|
|
|
scrollIfCan( board, xwe ); /* this must happen before we count blanks
|
|
since it invalidates squares */
|
|
|
|
/* This is freaking expensive!!!! PENDING FIXME Can't we start from
|
|
what's invalid rather than scanning the entire model every time
|
|
somebody dirties a single cell? */
|
|
model_listPlacedBlanks( model, board->selPlayer,
|
|
board->trayVisState == TRAY_REVEALED, &bq );
|
|
dragDropAppendBlank( board, &bq );
|
|
invalBlanksWithNeighbors( board, &bq );
|
|
|
|
/* figure out now, before clearing inval bits, if we'll need to draw
|
|
the arrow later */
|
|
if ( board->trayVisState == TRAY_REVEALED ) {
|
|
BoardArrow const* tmpArrow = &board->selInfo->boardArrow;
|
|
if ( tmpArrow->visible ) {
|
|
XP_U16 col = tmpArrow->col;
|
|
XP_U16 row = tmpArrow->row;
|
|
if ( INVAL_BIT_SET( board, col, row )
|
|
&& !cellOccupied( board, col, row, XP_TRUE ) ) {
|
|
arrow = tmpArrow;
|
|
}
|
|
}
|
|
}
|
|
|
|
nVisCols = model_numCols( model ) - board->zoomCount;
|
|
for ( row = vsd->offset; row <= vsd->lastVisible; ++row ) {
|
|
RowFlags rowFlags = board->redrawFlags[row];
|
|
if ( rowFlags != 0 ) {
|
|
RowFlags failedBits = 0;
|
|
for ( col = 0; col < nVisCols; ++col ) {
|
|
RowFlags colMask = 1 << (col + hsd->offset);
|
|
if ( 0 != (rowFlags & colMask) ) {
|
|
if ( !drawCell( board, xwe, col + hsd->offset,
|
|
row, XP_TRUE )) {
|
|
failedBits |= colMask;
|
|
allDrawn = XP_FALSE;
|
|
}
|
|
}
|
|
}
|
|
board->redrawFlags[row] = failedBits;
|
|
}
|
|
}
|
|
|
|
/* draw the blanks we skipped before */
|
|
for ( ii = 0; ii < bq.nBlanks; ++ii ) {
|
|
if ( !drawCell( board, xwe, bq.col[ii], bq.row[ii], XP_FALSE ) ) {
|
|
allDrawn = XP_FALSE;
|
|
}
|
|
}
|
|
|
|
if ( !!arrow ) {
|
|
XP_U16 col = arrow->col;
|
|
XP_U16 row = arrow->row;
|
|
XP_Rect arrowRect;
|
|
if ( getCellRect( board, col, row, &arrowRect ) ) {
|
|
XWBonusType bonus;
|
|
HintAtts hintAtts;
|
|
CellFlags flags = CELL_NONE;
|
|
bonus = model_getSquareBonus( model, xwe, col, row );
|
|
hintAtts = figureHintAtts( board, col, row );
|
|
#ifdef KEYBOARD_NAV
|
|
if ( cellFocused( board, col, row ) ) {
|
|
flags |= CELL_ISCURSOR;
|
|
}
|
|
#endif
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
flags |= flagsForCrosshairs( board, col, row );
|
|
#endif
|
|
|
|
draw_drawBoardArrow( board->draw, xwe, &arrowRect, bonus,
|
|
arrow->vert, hintAtts, flags );
|
|
}
|
|
}
|
|
|
|
/* I doubt the two of these can happen at the same time */
|
|
drawTradeWindowIf( board );
|
|
#ifdef POINTER_SUPPORT
|
|
drawDragTileIf( board, xwe );
|
|
#endif
|
|
draw_objFinished( board->draw, xwe, OBJ_BOARD, &board->boardBounds,
|
|
dfsFor( board, OBJ_BOARD ) );
|
|
|
|
board->needsDrawing = !allDrawn;
|
|
}
|
|
} /* drawBoard */
|
|
|
|
static XP_Bool
|
|
drawCell( BoardCtxt* board, XWEnv xwe, const XP_U16 col,
|
|
const XP_U16 row, XP_Bool skipBlanks )
|
|
{
|
|
XP_Bool success = XP_TRUE;
|
|
XP_Rect cellRect = {0};
|
|
Tile tile;
|
|
XP_Bool isBlank, isEmpty, pending = XP_FALSE;
|
|
XWBonusType bonus;
|
|
ModelCtxt* model = board->model;
|
|
const DictionaryCtxt* dict = model_getDictionary( model );
|
|
XP_U16 modelCol, modelRow;
|
|
|
|
if ( dict != NULL && getCellRect( board, col, row, &cellRect ) ) {
|
|
|
|
/* We want to invert EITHER the current pending tiles OR the most recent
|
|
* move. So if the tray is visible AND there are tiles missing from it,
|
|
* show them. Otherwise show the most recent move.
|
|
*/
|
|
XP_U16 selPlayer = board->selPlayer;
|
|
XP_U16 curCount = model_getCurrentMoveCount( model, selPlayer );
|
|
XP_Bool showPending = board->trayVisState == TRAY_REVEALED
|
|
&& curCount > 0;
|
|
|
|
flipIf( board, col, row, &modelCol, &modelRow );
|
|
|
|
/* This 'while' is only here so I can 'break' below */
|
|
while ( board->trayVisState == TRAY_HIDDEN ||
|
|
!rectContainsRect( &board->trayBounds, &cellRect ) ) {
|
|
XP_Bool recent = XP_FALSE;
|
|
XP_S16 owner = -1;
|
|
XP_Bitmaps bitmaps;
|
|
XP_Bitmaps* bptr = NULL;
|
|
const XP_UCHAR* textP = NULL;
|
|
HintAtts hintAtts;
|
|
CellFlags flags = CELL_NONE;
|
|
XP_Bool isOrigin;
|
|
XP_U16 value = 0;
|
|
|
|
isEmpty = !model_getTile( model, modelCol, modelRow, showPending,
|
|
selPlayer, &tile, &isBlank,
|
|
&pending, &recent );
|
|
if ( dragDropIsBeingDragged( board, col, row, &isOrigin ) ) {
|
|
flags |= isOrigin? CELL_DRAGSRC : CELL_DRAGCUR;
|
|
if ( isEmpty && !isOrigin ) {
|
|
dragDropTileInfo( board, &tile, &isBlank );
|
|
isEmpty = XP_FALSE;
|
|
}
|
|
showPending = pending = XP_TRUE;
|
|
}
|
|
|
|
if ( isEmpty ) {
|
|
isBlank = XP_FALSE;
|
|
flags |= CELL_ISEMPTY;
|
|
} else if ( isBlank && skipBlanks ) {
|
|
break;
|
|
} else {
|
|
Tile valTile = isBlank? dict_getBlankTile( dict ) : tile;
|
|
value = dict_getTileValue( dict, valTile );
|
|
|
|
if ( board->showColors ) {
|
|
owner = (XP_S16)model_getCellOwner( model, modelCol,
|
|
modelRow );
|
|
}
|
|
|
|
if ( dict_faceIsBitmap( dict, tile ) ) {
|
|
dict_getFaceBitmaps( dict, tile, &bitmaps );
|
|
bptr = &bitmaps;
|
|
}
|
|
textP = dict_getTileString( dict, tile );
|
|
}
|
|
bonus = model_getSquareBonus( model, xwe, col, row );
|
|
hintAtts = figureHintAtts( board, col, row );
|
|
|
|
if ( (col==board->star_row) && (row==board->star_row) ) {
|
|
flags |= CELL_ISSTAR;
|
|
}
|
|
if ( recent && !showPending ) {
|
|
flags |= CELL_RECENT;
|
|
} else if ( pending ) {
|
|
flags |= CELL_PENDING;
|
|
}
|
|
if ( isBlank ) {
|
|
flags |= CELL_ISBLANK;
|
|
}
|
|
#ifdef KEYBOARD_NAV
|
|
if ( cellFocused( board, col, row ) ) {
|
|
flags |= CELL_ISCURSOR;
|
|
}
|
|
#endif
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
flags |= flagsForCrosshairs( board, col, row );
|
|
#endif
|
|
|
|
success = draw_drawCell( board->draw, xwe, &cellRect, textP, bptr,
|
|
tile, value, owner, bonus, hintAtts,
|
|
flags );
|
|
#ifdef LOG_CELL_DRAW
|
|
XP_UCHAR buf[64];
|
|
XP_LOGF( "%s(col=%d, row=%d, flags=%s)=>%s", __func__, col, row,
|
|
formatFlags(buf, VSIZE(buf), flags), success?"true":"false" );
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
return success;
|
|
} /* drawCell */
|
|
|
|
#ifdef KEYBOARD_NAV
|
|
DrawFocusState
|
|
dfsFor( BoardCtxt* board, BoardObjectType obj )
|
|
{
|
|
DrawFocusState dfs;
|
|
if ( (board->focussed == obj) && !board->hideFocus ) {
|
|
if ( board->focusHasDived ) {
|
|
dfs = DFS_DIVED;
|
|
} else {
|
|
dfs = DFS_TOP;
|
|
}
|
|
} else {
|
|
dfs = DFS_NONE;
|
|
}
|
|
return dfs;
|
|
} /* dfsFor */
|
|
|
|
static XP_Bool
|
|
cellFocused( const BoardCtxt* board, XP_U16 col, XP_U16 row )
|
|
{
|
|
XP_Bool focussed = XP_FALSE;
|
|
const ScrollData* hsd = &board->sd[SCROLL_H];
|
|
const ScrollData* vsd = &board->sd[SCROLL_V];
|
|
if ( (board->focussed == OBJ_BOARD) && !board->hideFocus ) {
|
|
if ( board->focusHasDived ) {
|
|
if ( (col == board->selInfo->bdCursor.col)
|
|
&& (row == board->selInfo->bdCursor.row) ) {
|
|
focussed = XP_TRUE;
|
|
}
|
|
} else {
|
|
#ifdef PERIMETER_FOCUS
|
|
focussed = (col == hsd->offset)
|
|
|| (col == hsd->lastVisible)
|
|
|| (row == vsd->offset)
|
|
|| (row == vsd->lastVisible);
|
|
#else
|
|
focussed = XP_TRUE;
|
|
#endif
|
|
}
|
|
}
|
|
return focussed;
|
|
} /* cellFocused */
|
|
#endif
|
|
|
|
#ifdef POINTER_SUPPORT
|
|
static void
|
|
drawDragTileIf( BoardCtxt* board, XWEnv xwe )
|
|
{
|
|
if ( dragDropInProgress( board ) ) {
|
|
XP_U16 col, row;
|
|
if ( dragDropGetBoardTile( board, &col, &row ) ) {
|
|
XP_Rect rect;
|
|
Tile tile;
|
|
XP_Bool isBlank;
|
|
const XP_UCHAR* face;
|
|
XP_Bitmaps bitmaps;
|
|
XP_S16 value;
|
|
CellFlags flags;
|
|
|
|
getDragCellRect( board, col, row, &rect );
|
|
|
|
dragDropTileInfo( board, &tile, &isBlank );
|
|
|
|
face = getTileDrawInfo( board, tile, isBlank, &bitmaps,
|
|
&value );
|
|
|
|
flags = CELL_DRAGCUR;
|
|
if ( isBlank ) {
|
|
flags |= CELL_ISBLANK;
|
|
}
|
|
if ( board->hideValsInTray ) {
|
|
flags |= CELL_VALHIDDEN;
|
|
}
|
|
draw_drawTileMidDrag( board->draw, xwe, &rect, face,
|
|
bitmaps.nBitmaps > 0 ? &bitmaps : NULL,
|
|
value, board->selPlayer, flags );
|
|
}
|
|
}
|
|
} /* drawDragTileIf */
|
|
#endif
|
|
|
|
static XP_S16
|
|
sumRowHeights( const BoardCtxt* board, XP_U16 row1, XP_U16 row2 )
|
|
{
|
|
const ScrollData* vsd = &board->sd[SCROLL_V];
|
|
XP_S16 sign = row1 > row2 ? -1 : 1;
|
|
XP_S16 result, ii;
|
|
if ( sign < 0 ) {
|
|
XP_U16 tmp = row1;
|
|
row1 = row2;
|
|
row2 = tmp;
|
|
}
|
|
for ( result = 0, ii = row1; ii < row2; ++ii ) {
|
|
result += vsd->dims[ii];
|
|
}
|
|
return result * sign;
|
|
}
|
|
|
|
static void
|
|
scrollIfCan( BoardCtxt* board, XWEnv xwe )
|
|
{
|
|
ScrollData* vsd = &board->sd[SCROLL_V];
|
|
if ( vsd->offset != board->prevYScrollOffset ) {
|
|
XP_Rect scrollR = board->boardBounds;
|
|
XP_Bool scrolled;
|
|
XP_S16 dist;
|
|
|
|
#if defined KEYBOARD_NAV && defined PERIMETER_FOCUS
|
|
if ( (board->focussed == OBJ_BOARD)
|
|
&& !board->focusHasDived
|
|
&& !board->hideFocus ) {
|
|
invalOldPerimeter( board );
|
|
}
|
|
#endif
|
|
invalSelTradeWindow( board );
|
|
|
|
dist = sumRowHeights( board, board->prevYScrollOffset, vsd->offset );
|
|
scrolled = draw_vertScrollBoard( board->draw, xwe, &scrollR, dist,
|
|
dfsFor( board, OBJ_BOARD ) );
|
|
|
|
if ( scrolled ) {
|
|
/* inval the rows that have been scrolled into view. I'm cheating
|
|
making the client figure the inval rect, but Palm's the first
|
|
client and it does it so well.... */
|
|
invalCellsUnderRect( board, &scrollR );
|
|
} else {
|
|
board_invalAll( board );
|
|
}
|
|
board->prevYScrollOffset = vsd->offset;
|
|
}
|
|
} /* scrollIfCan */
|
|
|
|
#ifdef XWFEATURE_MINIWIN
|
|
static void
|
|
drawTradeWindowIf( BoardCtxt* board )
|
|
{
|
|
if ( board->tradingMiniWindowInvalid &&
|
|
TRADE_IN_PROGRESS(board) && board->trayVisState == TRAY_REVEALED ) {
|
|
MiniWindowStuff* stuff;
|
|
|
|
makeMiniWindowForTrade( board );
|
|
|
|
stuff = &board->miniWindowStuff[MINIWINDOW_TRADING];
|
|
draw_drawMiniWindow( board->draw, stuff->text,
|
|
&stuff->rect, (void**)NULL );
|
|
|
|
board->tradingMiniWindowInvalid = XP_FALSE;
|
|
}
|
|
} /* drawTradeWindowIf */
|
|
#endif
|
|
|
|
XP_Bool
|
|
board_draw( BoardCtxt* board, XWEnv xwe )
|
|
{
|
|
if ( !!board->draw && board->boardBounds.width > 0 ) {
|
|
if ( draw_beginDraw( board->draw, xwe ) ) {
|
|
|
|
drawScoreBoard( board, xwe );
|
|
drawTray( board, xwe );
|
|
drawBoard( board, xwe );
|
|
|
|
draw_endDraw( board->draw, xwe );
|
|
}
|
|
}
|
|
return !board->needsDrawing && 0 == board->trayInvalBits;
|
|
} /* board_draw */
|
|
|
|
#ifdef LOG_CELL_DRAW
|
|
static XP_UCHAR* formatFlags( XP_UCHAR* buf, XP_U16 len, CellFlags flags )
|
|
{
|
|
XP_U16 used = XP_SNPRINTF( buf, len, "0x%x ", flags );
|
|
for ( CellFlags flag = 1; flag < CELL_ALL; flag <<= 1 ) {
|
|
XP_UCHAR* str = NULL;
|
|
if ( 0 != (flag & flags) ) {
|
|
switch (flag) {
|
|
case CELL_ISBLANK: str = "BLNK"; break;
|
|
case CELL_PENDING: str = "PEND"; break;
|
|
case CELL_RECENT: str = "RCNT"; break;
|
|
case CELL_ISEMPTY: str = "MPTY"; break;
|
|
case CELL_ISCURSOR: str = "CURS"; break;
|
|
case CELL_VALHIDDEN: str = "HIDN"; break;
|
|
case CELL_DRAGSRC:
|
|
case CELL_DRAGCUR:
|
|
case CELL_CROSSVERT:
|
|
case CELL_CROSSHOR:
|
|
case CELL_ISSTAR:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( NULL != str ) {
|
|
used += XP_SNPRINTF( buf + used, len - used, "%s;", str );
|
|
}
|
|
}
|
|
return buf;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef CPLUS
|
|
}
|
|
#endif
|