mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-22 07:28:16 +01:00
7fe249ad8d
Rather than have a tile revert to its original location if it's dropped where it can't be (on an occupied tile or outside the board), put it in the last place it could have been. Do that by only updating cur when it's to a legal location, and then relocating to cur when drag ends.
813 lines
26 KiB
C
813 lines
26 KiB
C
/* -*-mode: C; fill-column: 78; compile-command: "cd ../linux && make MEMDEBUG=TRUE -j3"; -*- */
|
|
/*
|
|
* Copyright 1997 - 2010 by Eric House (xwords@eehouse.org). All rights
|
|
* reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#ifdef CPLUS
|
|
extern "C" {
|
|
#endif
|
|
|
|
#include "dragdrpp.h"
|
|
#include "game.h"
|
|
#include "dbgutil.h"
|
|
|
|
/* How many squares must scroll gesture take in to be recognized. */
|
|
#ifndef SCROLL_DRAG_THRESHHOLD
|
|
# define SCROLL_DRAG_THRESHHOLD 1
|
|
#endif
|
|
|
|
static XP_Bool dragDropContinueImpl( BoardCtxt* board, XWEnv xwe, XP_U16 xx,
|
|
XP_U16 yy, BoardObjectType* onWhichP );
|
|
static void invalDragObjRange( BoardCtxt* board, const DragObjInfo* from,
|
|
const DragObjInfo* to );
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static void invalHintRectDiffs( BoardCtxt* board, const DragObjInfo* cur,
|
|
const DragObjInfo* nxt );
|
|
static void setLimitsFrom( const BoardCtxt* board, BdHintLimits* limits );
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
static XP_Bool crosshairs_set( BoardCtxt* board, XP_S16 col, XP_S16 row,
|
|
XP_Bool clearOnMove );
|
|
static void crosshairs_clear( BoardCtxt* board );
|
|
#else
|
|
# define crosshairs_set( board, col, row, com ) XP_FALSE;
|
|
# define crosshairs_clear( board )
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
static void log_ds( const DragState* ds, const char* proc, const char* comment );
|
|
# define LOG_DS(DS, COMMENT) log_ds( (DS), __func__, (COMMENT) )
|
|
#else
|
|
# define LOG_DS(DS, COMMENT)
|
|
#endif
|
|
|
|
static void startScrollTimerIf( BoardCtxt* board, XWEnv xwe );
|
|
|
|
XP_Bool
|
|
dragDropInProgress( const BoardCtxt* board )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
/* LOG_RETURNF( "%d", ds->dragInProgress ); */
|
|
return ds->dtype != DT_NONE;
|
|
} /* dragDropInProgress */
|
|
|
|
XP_Bool
|
|
dragDropHasMoved( const BoardCtxt* board )
|
|
{
|
|
XP_Bool moved = dragDropInProgress( board );
|
|
if ( moved ) {
|
|
if ( board->dragState.didMove ) {
|
|
/* something was dragged; do nothing */
|
|
} else {
|
|
const DragState* ds = &board->dragState;
|
|
moved = ds->cellChanged; /* did non-drag movement happen? */
|
|
}
|
|
}
|
|
return moved;
|
|
} /* dragDropHasMoved */
|
|
|
|
static void
|
|
clearDS( DragState* ds )
|
|
{
|
|
XP_MEMSET( ds, 0, sizeof(*ds) );
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
ds->crosshairs.col = ds->crosshairs.row = -1;
|
|
#endif
|
|
}
|
|
|
|
static XP_Bool
|
|
ddStartBoard( BoardCtxt* board, XWEnv xwe, XP_U16 xx, XP_U16 yy )
|
|
{
|
|
DragState* ds = &board->dragState;
|
|
XP_Bool trayVisible;
|
|
XP_U16 col, row;
|
|
|
|
#ifdef DEBUG
|
|
XP_Bool found =
|
|
#endif
|
|
coordToCell( board, xx, yy, &col, &row );
|
|
XP_ASSERT( found );
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
if ( !board->hideCrosshairs ) {
|
|
(void)crosshairs_set( board, col, row, XP_FALSE );
|
|
}
|
|
#endif
|
|
|
|
trayVisible = board->trayVisState == TRAY_REVEALED;
|
|
if ( trayVisible && holdsPendingTile( board, col, row ) ) {
|
|
XP_U16 modelc, modelr;
|
|
|
|
ds->dtype = DT_TILE;
|
|
flipIf( board, col, row, &modelc, &modelr );
|
|
|
|
#ifdef DEBUG
|
|
found =
|
|
#endif
|
|
model_getTile( board->model, modelc, modelr, XP_TRUE,
|
|
board->selPlayer, &ds->tile, &ds->isBlank,
|
|
NULL, NULL );
|
|
XP_ASSERT( found );
|
|
} else {
|
|
/* If we're not dragging a tile, we can either drag the board (scroll)
|
|
or work on hint regions. Sometimes scrolling isn't possible.
|
|
Sometimes hint dragging is disabled. But if both are possible,
|
|
then the alt key determines it. I figure scrolling will be more
|
|
common than hint dragging when both are possible, but you can turn
|
|
hint dragging off, so if it's on that's probably what you want. */
|
|
ScrollData* vsd = &board->sd[SCROLL_V];
|
|
XP_Bool canScroll = vsd->lastVisible < model_numRows(board->model);
|
|
if ( 0 ) {
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
} else if ( !board->gi->hintsNotAllowed && board->gi->allowHintRect
|
|
&& trayVisible ) {
|
|
if ( !util_altKeyDown(board->util, xwe) ) {
|
|
ds->dtype = DT_HINTRGN;
|
|
} else if ( canScroll ) {
|
|
ds->dtype = DT_BOARD;
|
|
}
|
|
#endif
|
|
} else if ( canScroll ) {
|
|
ds->dtype = DT_BOARD;
|
|
}
|
|
}
|
|
ds->start.u.board.col = col;
|
|
ds->start.u.board.row = row;
|
|
|
|
return ds->dtype != DT_NONE;
|
|
} /* ddStartBoard */
|
|
|
|
static XP_Bool
|
|
ddStartTray( BoardCtxt* board, XP_U16 x, XP_U16 y )
|
|
{
|
|
XP_Bool canDrag;
|
|
DragState* ds = &board->dragState;
|
|
|
|
XP_Bool onDivider;
|
|
XP_S16 index = pointToTileIndex( board, x, y, &onDivider );
|
|
canDrag = onDivider || index >= 0;
|
|
if ( canDrag ) {
|
|
if ( onDivider ) {
|
|
board->dividerInvalid = XP_TRUE;
|
|
ds->start.u.tray.index =
|
|
model_getDividerLoc( board->model, board->selPlayer );
|
|
|
|
ds->dtype = DT_DIVIDER;
|
|
} else {
|
|
Tile tile;
|
|
tile = model_getPlayerTile( board->model, board->selPlayer, index );
|
|
ds->isBlank =
|
|
tile == dict_getBlankTile( model_getDictionary(board->model) );
|
|
ds->tile = tile;
|
|
|
|
ds->start.u.tray.index = index;
|
|
|
|
/* during drag the moving tile is drawn as selected, so inval
|
|
currently selected tile. */
|
|
board_invalTrayTiles( board, board->selInfo->traySelBits );
|
|
|
|
ds->dtype = DT_TILE;
|
|
}
|
|
}
|
|
|
|
return canDrag;
|
|
} /* ddStartTray */
|
|
|
|
/* x and y, in board coordinates (not model, should the board be flipped), are
|
|
* already col,row (cells) in the board case, but real x/y (pixels) on the
|
|
* tray.
|
|
*/
|
|
XP_Bool
|
|
dragDropStart( BoardCtxt* board, XWEnv xwe, BoardObjectType obj, XP_U16 xx, XP_U16 yy )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
DragState* ds = &board->dragState;
|
|
if ( dragDropInProgress(board) ) {
|
|
XP_LOGF( "warning: starting drag while dragDropInProgress() true" );
|
|
}
|
|
clearDS( ds );
|
|
|
|
ds->start.obj = obj;
|
|
|
|
if ( OBJ_BOARD == obj ) {
|
|
result = ddStartBoard( board, xwe, xx, yy );
|
|
} else if ( OBJ_TRAY == obj ) {
|
|
result = ddStartTray( board, xx, yy );
|
|
} else {
|
|
XP_ASSERT(0);
|
|
}
|
|
|
|
if ( result ) {
|
|
ds->cur = ds->start;
|
|
invalDragObj( board, &ds->start );
|
|
startScrollTimerIf( board, xwe );
|
|
}
|
|
|
|
return result;
|
|
} /* dragDropStart */
|
|
|
|
/* We're potentially dragging. If we're leaving one object, inval it.
|
|
* If we're entering another, inval it. Track where we are so we don't inval
|
|
* again. If we didn't really move, don't inval anything!
|
|
*
|
|
* Our overriding concern must be to preserve data integrity in the face of
|
|
* buggy OSes that may drop events (esp. MouseUp). So never own a tile: keep
|
|
* it attached either to the tray or the board. If the state of the board is
|
|
* to change as dragging happens, that's for the board (view) to display, not
|
|
* the model. So drawing has to be aware of drag-and-drop, but we don't
|
|
* change the model until the pen's actually lifted. If we never get the
|
|
* pen-up event at the worst the board's drawn incorrectly for a bit.
|
|
*
|
|
* Exception: since divider location is in the board rather than the model
|
|
* just change it every time we can.
|
|
*/
|
|
XP_Bool
|
|
dragDropContinue( BoardCtxt* board, XWEnv xwe, XP_U16 xx, XP_U16 yy )
|
|
{
|
|
XP_ASSERT( dragDropInProgress(board) );
|
|
|
|
return dragDropContinueImpl( board, xwe, xx, yy, NULL );
|
|
}
|
|
|
|
XP_Bool
|
|
dragDropEnd( BoardCtxt* board, XWEnv xwe, XP_U16 xx, XP_U16 yy, XP_Bool* draggedP )
|
|
{
|
|
DragState* ds = &board->dragState;
|
|
BoardObjectType newObj;
|
|
|
|
LOG_DS( ds, "starting" );
|
|
|
|
XP_ASSERT( dragDropInProgress(board) );
|
|
|
|
(void)dragDropContinueImpl( board, xwe, xx, yy, &newObj );
|
|
if ( !!draggedP ) {
|
|
*draggedP = ds->didMove;
|
|
}
|
|
|
|
/* If we've dropped on something, put the tile there! Since we
|
|
don't remove it from its earlier location until it's dropped,
|
|
we need to specify where it's coming from. */
|
|
if ( ds->dtype == DT_DIVIDER ) {
|
|
board->dividerInvalid = XP_TRUE;
|
|
/* nothing to do */
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
} else if ( ds->dtype == DT_HINTRGN ) {
|
|
if ( OBJ_BOARD == newObj && ds->didMove ) {
|
|
XP_Bool makeActive = ds->start.u.board.row <= ds->cur.u.board.row;
|
|
board->selInfo->hasHintRect = makeActive;
|
|
if ( makeActive ) {
|
|
setLimitsFrom( board, &board->selInfo->limits );
|
|
} else {
|
|
invalHintRectDiffs( board, &ds->cur, NULL );
|
|
}
|
|
board_resetEngine( board );
|
|
} else {
|
|
/* return it to whatever it was */
|
|
invalHintRectDiffs( board, &ds->cur, NULL );
|
|
invalCurHintRect( board, board->selPlayer );
|
|
}
|
|
#endif
|
|
} else if ( ds->dtype == DT_BOARD ) {
|
|
/* do nothing */
|
|
} else if ( ds->dtype == DT_TILE ) {
|
|
XP_U16 mod_startc, mod_startr;
|
|
|
|
flipIf( board, ds->start.u.board.col, ds->start.u.board.row,
|
|
&mod_startc, &mod_startr );
|
|
|
|
if ( ds->cur.obj == OBJ_TRAY ) {
|
|
if ( ds->start.obj == OBJ_BOARD ) { /* board->tray is pending */
|
|
model_moveBoardToTray( board->model, xwe, board->selPlayer,
|
|
mod_startc, mod_startr,
|
|
ds->cur.u.tray.index );
|
|
} else {
|
|
model_moveTileOnTray( board->model, board->selPlayer,
|
|
ds->start.u.tray.index,
|
|
ds->cur.u.tray.index );
|
|
}
|
|
} else if ( ds->cur.obj == OBJ_BOARD
|
|
/* PENDING: This test shouldn't be necessary. It's tied up in the
|
|
pen timer triggering a words-under-here listing when it
|
|
shouldn't */
|
|
&& !cellOccupied( board, ds->cur.u.board.col,
|
|
ds->cur.u.board.row, XP_TRUE ) ) {
|
|
if ( ds->start.obj == OBJ_TRAY ) {
|
|
/* moveTileToBoard flips its inputs */
|
|
(void)moveTileToBoard( board, xwe, ds->cur.u.board.col,
|
|
ds->cur.u.board.row,
|
|
ds->start.u.tray.index, EMPTY_TILE );
|
|
} else if ( ds->start.obj == OBJ_BOARD ) {
|
|
XP_U16 mod_curc, mod_curr;
|
|
flipIf( board, ds->cur.u.board.col, ds->cur.u.board.row,
|
|
&mod_curc, &mod_curr );
|
|
if ( model_moveTileOnBoard( board->model, xwe, board->selPlayer,
|
|
mod_startc, mod_startr, mod_curc,
|
|
mod_curr ) ) {
|
|
/* inval points tile in case score changed */
|
|
board_invalTrayTiles( board, 1 << (MAX_TRAY_TILES-1) );
|
|
}
|
|
}
|
|
} else {
|
|
/* We're returning it to start, so will be re-inserted in tray */
|
|
if ( OBJ_TRAY == ds->start.obj ) {
|
|
board_invalTrayTiles( board, ALLTILES );
|
|
}
|
|
}
|
|
|
|
/* These may change appearance, e.g. from big tile to dropped cell. */
|
|
invalDragObj( board, &ds->cur );
|
|
invalDragObj( board, &ds->start );
|
|
}
|
|
|
|
crosshairs_clear( board );
|
|
ds->dtype = DT_NONE;
|
|
|
|
return XP_TRUE;
|
|
} /* dragDropEnd */
|
|
|
|
#ifdef XWFEATURE_RAISETILE
|
|
XP_Bool
|
|
dragDropSetAdd( BoardCtxt* board, XWEnv xwe )
|
|
{
|
|
DragState* ds = &board->dragState;
|
|
XP_Bool draw;
|
|
draw = dragDropInProgress(board) && !dragDropIsDividerDrag( board )
|
|
&& !dragDropHasMoved( board );
|
|
if ( draw ) {
|
|
if ( draw ) {
|
|
ds->yyAdd = (board->trayBounds.height * 2) / 3;
|
|
draw = dragDropContinueImpl( board, xwe, board->penDownX,
|
|
board->penDownY, NULL );
|
|
}
|
|
}
|
|
return draw;
|
|
}
|
|
#endif
|
|
|
|
XP_Bool
|
|
dragDropGetBoardTile( const BoardCtxt* board, XP_U16* col, XP_U16* row )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
XP_Bool found = ds->dtype == DT_TILE && ds->cur.obj == OBJ_BOARD;
|
|
if ( found ) {
|
|
*col = ds->cur.u.board.col;
|
|
*row = ds->cur.u.board.row;
|
|
}
|
|
return found;
|
|
}
|
|
|
|
XP_Bool
|
|
dragDropIsBeingDragged( const BoardCtxt* board, XP_U16 col, XP_U16 row,
|
|
XP_Bool* isOrigin )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
XP_Bool result = ds->dtype == DT_TILE && ds->cur.obj == OBJ_BOARD;
|
|
if ( result && !!isOrigin ) {
|
|
const DragState* ds = &board->dragState;
|
|
if ( (ds->cur.obj == OBJ_BOARD) && (ds->cur.u.board.col == col)
|
|
&& (ds->cur.u.board.row == row) ) {
|
|
*isOrigin = XP_FALSE;
|
|
} else if ( (ds->start.obj == OBJ_BOARD)
|
|
&& (ds->start.u.board.col == col)
|
|
&& (ds->start.u.board.row == row) ) {
|
|
*isOrigin = XP_TRUE;
|
|
} else {
|
|
result = XP_FALSE;
|
|
}
|
|
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void
|
|
dragDropAppendBlank( const BoardCtxt* board, BlankQueue* bqp )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
if ( ds->dtype == DT_TILE && ds->cur.obj == OBJ_BOARD ) {
|
|
if ( ds->isBlank ) {
|
|
bqp->col[bqp->nBlanks] = ds->cur.u.board.col;
|
|
bqp->row[bqp->nBlanks] = ds->cur.u.board.row;
|
|
++bqp->nBlanks;
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
dragDropGetTrayChanges( const BoardCtxt* board, XP_U16* rmvdIndx,
|
|
XP_U16* addedIndx )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
*addedIndx = *rmvdIndx = MAX_TRAY_TILES; /* too big means ignore me */
|
|
if ( ds->dtype == DT_TILE ) {
|
|
if ( OBJ_TRAY == ds->start.obj ) {
|
|
*rmvdIndx = ds->start.u.tray.index;
|
|
}
|
|
if ( OBJ_TRAY == ds->cur.obj ) {
|
|
*addedIndx = ds->cur.u.tray.index;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
XP_Bool
|
|
dragDropGetHintLimits( const BoardCtxt* board, BdHintLimits* limits )
|
|
{
|
|
XP_Bool result = board->dragState.dtype == DT_HINTRGN;
|
|
if ( result ) {
|
|
setLimitsFrom( board, limits );
|
|
}
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
XP_Bool
|
|
dragDropIsDividerDrag( const BoardCtxt* board )
|
|
{
|
|
return board->dragState.dtype == DT_DIVIDER;
|
|
}
|
|
|
|
void
|
|
dragDropTileInfo( const BoardCtxt* board, Tile* tile, XP_Bool* isBlank )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
XP_ASSERT( dragDropInProgress( board ) );
|
|
XP_ASSERT( ds->dtype == DT_TILE );
|
|
XP_ASSERT ( OBJ_BOARD == ds->start.obj || OBJ_TRAY == ds->start.obj );
|
|
*tile = ds->tile;
|
|
*isBlank = ds->isBlank;
|
|
}
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static void
|
|
invalHintRectDiffs( BoardCtxt* board, const DragObjInfo* cur,
|
|
const DragObjInfo* nxt )
|
|
{
|
|
XP_U16 startCol = board->dragState.start.u.board.col;
|
|
XP_U16 startRow = board->dragState.start.u.board.row;
|
|
|
|
/* These two regions will generally have close to 50% of their borders in
|
|
common. Try not to inval what needn't be inval'd. But at the moment
|
|
performance seems good enough without adding the complexity and new
|
|
bugs...
|
|
|
|
The challenge in doing a smarter diff is that some squares need to be
|
|
invalidated even if they're part of the borders of both limits rects,
|
|
in particular if one is a corner of one and just a side of another.
|
|
One simple but expensive way of accounting for this would be to call
|
|
figureHintAtts() on each square in the borders of both rects and
|
|
invalidate when the hintAttributes aren't the same for both. That
|
|
misses an opportunity to avoid doing any calculations on those border
|
|
squares that clearly haven't changed at all.
|
|
*/
|
|
|
|
invalCellRegion( board, startCol, startRow, cur->u.board.col,
|
|
cur->u.board.row );
|
|
if ( !!nxt ) {
|
|
BdHintLimits limits;
|
|
setLimitsFrom( board, &limits );
|
|
invalCellRegion( board, startCol, startRow, nxt->u.board.col,
|
|
nxt->u.board.row );
|
|
}
|
|
} /* invalHintRectDiffs */
|
|
#endif
|
|
|
|
static XP_Bool
|
|
dragDropContinueImpl( BoardCtxt* board, XWEnv xwe, XP_U16 xx, XP_U16 yy,
|
|
BoardObjectType* onWhichP )
|
|
{
|
|
XP_Bool moving = XP_FALSE;
|
|
DragObjInfo newInfo;
|
|
DragState* ds = &board->dragState;
|
|
XP_Bool draw = XP_FALSE;
|
|
|
|
#ifdef XWFEATURE_RAISETILE
|
|
yy -= ds->yyAdd;
|
|
#endif
|
|
|
|
(void)pointOnSomething( board, xx, yy, &newInfo.obj );
|
|
if ( !!onWhichP ) {
|
|
*onWhichP = newInfo.obj;
|
|
}
|
|
XP_Bool isLegalBoardDrag = XP_FALSE;
|
|
if ( newInfo.obj == OBJ_BOARD
|
|
&& coordToCell( board, xx, yy, &newInfo.u.board.col,
|
|
&newInfo.u.board.row )
|
|
&& !cellOccupied( board, newInfo.u.board.col,
|
|
newInfo.u.board.row, XP_TRUE ) ) {
|
|
isLegalBoardDrag = XP_TRUE;
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
if ( !board->hideCrosshairs ) {
|
|
draw = crosshairs_set( board, newInfo.u.board.col,
|
|
newInfo.u.board.row,
|
|
DT_TILE != ds->dtype );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if ( ds->dtype == DT_DIVIDER ) {
|
|
if ( OBJ_TRAY == newInfo.obj ) {
|
|
XP_U16 newloc;
|
|
XP_U16 scale = board->trayScaleH;
|
|
xx -= board->trayBounds.left;
|
|
newloc = xx / scale;
|
|
if ( (xx % scale) > ((scale+board->dividerWidth)/2)) {
|
|
++newloc;
|
|
}
|
|
moving = dividerMoved( board, newloc );
|
|
}
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
} else if ( ds->dtype == DT_HINTRGN && newInfo.obj != OBJ_BOARD ) {
|
|
/* do nothing */
|
|
#endif
|
|
} else if ( ds->dtype == DT_BOARD ) {
|
|
if ( newInfo.obj == OBJ_BOARD ) {
|
|
XP_S16 diff = newInfo.u.board.col - ds->cur.u.board.col;
|
|
if ( !ds->cellChanged && 0 != diff ) {
|
|
ds->cellChanged = XP_TRUE;
|
|
}
|
|
diff /= SCROLL_DRAG_THRESHHOLD;
|
|
moving = adjustXOffset( board, diff );
|
|
|
|
diff = newInfo.u.board.row - ds->cur.u.board.row;
|
|
if ( !ds->cellChanged && 0 != diff ) {
|
|
ds->cellChanged = XP_TRUE;
|
|
}
|
|
diff /= SCROLL_DRAG_THRESHHOLD;
|
|
moving = adjustYOffset( board, xwe, diff ) || moving;
|
|
}
|
|
} else if ( ds->dtype == DT_TILE ) {
|
|
if ( newInfo.obj == OBJ_BOARD && isLegalBoardDrag ) {
|
|
moving = (newInfo.u.board.col != ds->cur.u.board.col)
|
|
|| (newInfo.u.board.row != ds->cur.u.board.row)
|
|
|| (OBJ_TRAY == ds->cur.obj);
|
|
} else if ( newInfo.obj == OBJ_TRAY ) {
|
|
XP_Bool onDivider;
|
|
XP_S16 index = pointToTileIndex( board, xx, yy, &onDivider );
|
|
if ( !onDivider ) {
|
|
if ( index < 0 ) { /* negative means onto empty part of
|
|
tray. Force left. */
|
|
index = model_getNumTilesInTray( board->model,
|
|
board->selPlayer );
|
|
if ( OBJ_TRAY == ds->start.obj ) {
|
|
--index; /* dragging right into space */
|
|
}
|
|
}
|
|
moving = (OBJ_BOARD == ds->cur.obj)
|
|
|| (index != ds->cur.u.tray.index);
|
|
if ( moving ) {
|
|
newInfo.u.tray.index = index;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( moving ) {
|
|
if ( ds->dtype == DT_TILE ) {
|
|
invalDragObjRange( board, &ds->cur, &newInfo );
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
} else if ( ds->dtype == DT_HINTRGN ) {
|
|
invalHintRectDiffs( board, &ds->cur, &newInfo );
|
|
if ( !ds->didMove ) { /* first time through */
|
|
invalCurHintRect( board, board->selPlayer );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
XP_MEMCPY( &ds->cur, &newInfo, sizeof(ds->cur) );
|
|
startScrollTimerIf( board, xwe );
|
|
}
|
|
} else {
|
|
XP_ASSERT( 0 );
|
|
}
|
|
|
|
if ( moving ) {
|
|
draw = XP_TRUE;
|
|
if ( !ds->didMove ) {
|
|
/* This is the first time we've moved!!! Kill any future timers,
|
|
and if there's a window up kill it.*/
|
|
board->penTimerFired = XP_FALSE;
|
|
#ifdef XWFEATURE_MINIWIN
|
|
if ( valHintMiniWindowActive( board ) ) {
|
|
hideMiniWindow( board, XP_TRUE, MINIWINDOW_VALHINT );
|
|
}
|
|
#endif
|
|
ds->didMove = XP_TRUE;
|
|
}
|
|
}
|
|
|
|
return draw;
|
|
} /* dragDropContinueImpl */
|
|
|
|
static void
|
|
invalDragObjRange( BoardCtxt* board, const DragObjInfo* from,
|
|
const DragObjInfo* to )
|
|
{
|
|
invalDragObj( board, from );
|
|
if ( NULL != to ) {
|
|
invalDragObj( board, to );
|
|
|
|
if ( (OBJ_TRAY == from->obj) && (OBJ_TRAY == to->obj) ) {
|
|
invalTrayTilesBetween( board, from->u.tray.index,
|
|
to->u.tray.index );
|
|
} else if ( OBJ_TRAY == from->obj ) {
|
|
invalTrayTilesAbove( board, from->u.tray.index );
|
|
} else if ( OBJ_TRAY == to->obj ) {
|
|
invalTrayTilesAbove( board, to->u.tray.index );
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static void
|
|
setLimitsFrom( const BoardCtxt* board, BdHintLimits* limits )
|
|
{
|
|
const DragState* ds = &board->dragState;
|
|
limits->left = XP_MIN( ds->start.u.board.col, ds->cur.u.board.col );
|
|
limits->right = XP_MAX( ds->start.u.board.col, ds->cur.u.board.col );
|
|
limits->top = XP_MIN( ds->start.u.board.row, ds->cur.u.board.row );
|
|
limits->bottom = XP_MAX( ds->start.u.board.row, ds->cur.u.board.row );
|
|
}
|
|
#endif
|
|
|
|
static XP_Bool
|
|
scrollTimerProc( void* closure, XWEnv xwe, XWTimerReason XP_UNUSED_DBG(why) )
|
|
{
|
|
XP_Bool draw = XP_FALSE;
|
|
BoardCtxt* board = (BoardCtxt*)closure;
|
|
DragState* ds = &board->dragState;
|
|
XP_ASSERT( why == TIMER_PENDOWN );
|
|
|
|
if ( ds->scrollTimerSet ) {
|
|
XP_S16 changeX = 0;
|
|
XP_S16 changeY = 0;
|
|
ds->scrollTimerSet = XP_FALSE;
|
|
XP_Bool canScroll = onBorderCanScroll( board, SCROLL_H,
|
|
ds->cur.u.board.col, &changeX );
|
|
canScroll = onBorderCanScroll( board, SCROLL_V, ds->cur.u.board.row,
|
|
&changeY ) || canScroll;
|
|
if ( canScroll ) {
|
|
invalDragObj( board, &ds->cur );
|
|
if ( 0 != changeX ) {
|
|
ds->cur.u.board.col += (changeX >0 ? 1 : -1);
|
|
}
|
|
if ( 0 != changeY ) {
|
|
ds->cur.u.board.row += (changeY >0 ? 1 : -1);
|
|
}
|
|
if ( scrollIntoView( board, xwe, ds->cur.u.board.col,
|
|
ds->cur.u.board.row ) ) {
|
|
board_draw( board, xwe ); /* may fail, e.g. on wince */
|
|
startScrollTimerIf( board, xwe );
|
|
draw = XP_TRUE;
|
|
}
|
|
}
|
|
}
|
|
return draw;
|
|
} /* scrollTimerProc */
|
|
|
|
static void
|
|
startScrollTimerIf( BoardCtxt* board, XWEnv xwe )
|
|
{
|
|
DragState* ds = &board->dragState;
|
|
|
|
if ( (ds->dtype == DT_TILE) && (ds->cur.obj == OBJ_BOARD) ) {
|
|
XP_S16 ignore;
|
|
if ( onBorderCanScroll( board, SCROLL_H, ds->cur.u.board.col, &ignore )
|
|
|| onBorderCanScroll( board, SCROLL_V, ds->cur.u.board.row,
|
|
&ignore ) ) {
|
|
util_setTimer( board->util, xwe, TIMER_PENDOWN, 0,
|
|
scrollTimerProc, (void*) board );
|
|
ds->scrollTimerSet = XP_TRUE;
|
|
} else {
|
|
/* ignore if we've moved off */
|
|
ds->scrollTimerSet = XP_FALSE;
|
|
}
|
|
}
|
|
} /* startScrollTimerIf */
|
|
|
|
#ifdef XWFEATURE_CROSSHAIRS
|
|
void
|
|
dragDropInCrosshairs( const BoardCtxt* board, XP_U16 col, XP_U16 row,
|
|
XP_Bool* inHor, XP_Bool* inVert )
|
|
{
|
|
|
|
if ( dragDropInProgress( board ) ) {
|
|
const DragState* ds = &board->dragState;
|
|
*inHor = ds->crosshairs.row == row;
|
|
*inVert = ds->crosshairs.col == col;
|
|
} else {
|
|
*inHor = *inVert = XP_FALSE;
|
|
}
|
|
} /* dragDropInCrosshairs */
|
|
|
|
static XP_Bool
|
|
crosshairs_set( BoardCtxt* board, XP_S16 col, XP_S16 row, XP_Bool clearOnMove )
|
|
{
|
|
XP_Bool changed;
|
|
DragState* ds = &board->dragState;
|
|
|
|
XP_Bool colMoving = ds->crosshairs.col != col;
|
|
XP_Bool rowMoving = ds->crosshairs.row != row;
|
|
if ( clearOnMove && (colMoving || rowMoving) ) {
|
|
changed = XP_TRUE;
|
|
crosshairs_clear( board );
|
|
} else {
|
|
changed = XP_FALSE;
|
|
if ( colMoving ) {
|
|
if ( ds->crosshairs.col >= 0 ) {
|
|
invalCol( board, ds->crosshairs.col );
|
|
}
|
|
if ( col >= 0 ) {
|
|
invalCol( board, col );
|
|
}
|
|
ds->crosshairs.col = col;
|
|
changed = XP_TRUE;
|
|
}
|
|
if ( rowMoving ) {
|
|
if ( ds->crosshairs.row >= 0 ) {
|
|
invalRow( board, ds->crosshairs.row );
|
|
}
|
|
if ( row >= 0 ) {
|
|
invalRow( board, row );
|
|
}
|
|
ds->crosshairs.row = row;
|
|
changed = XP_TRUE;
|
|
}
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
static void
|
|
crosshairs_clear( BoardCtxt* board )
|
|
{
|
|
crosshairs_set( board, -1, -1, XP_FALSE );
|
|
}
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
|
|
const XP_UCHAR*
|
|
DragTypeToStr( DragType typ )
|
|
{
|
|
#define CASE_STR(typ) case (typ): return #typ
|
|
switch( typ ) {
|
|
CASE_STR(DT_NONE);
|
|
CASE_STR(DT_DIVIDER);
|
|
CASE_STR(DT_TILE);
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
CASE_STR(DT_HINTRGN);
|
|
#endif
|
|
CASE_STR(DT_BOARD);
|
|
}
|
|
#undef CASE_STR
|
|
XP_ASSERT(0);
|
|
return NULL;
|
|
}
|
|
|
|
static const XP_UCHAR*
|
|
formatDragObj( const DragObjInfo* doi, XP_UCHAR buf[], XP_U16 len )
|
|
{
|
|
switch( doi->obj ) {
|
|
case OBJ_BOARD:
|
|
XP_SNPRINTF( buf, len, "board: {%d, %d}", doi->u.board.col, doi->u.board.row );
|
|
break;
|
|
case OBJ_TRAY:
|
|
XP_SNPRINTF( buf, len, "tray: {%d}", doi->u.tray.index );
|
|
break;
|
|
case OBJ_NONE:
|
|
XP_SNPRINTF( buf, len, "NONE" );
|
|
break;
|
|
default:
|
|
XP_ASSERT(0);
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
static void
|
|
log_ds( const DragState* ds, const char* proc, const char* comment )
|
|
{
|
|
XP_UCHAR startBuf[64];
|
|
XP_UCHAR curBuf[64];
|
|
XP_LOGF( "%s: DragState from %s(): {type: %s, start: %s, cur: %s}",
|
|
comment, proc, DragTypeToStr(ds->dtype),
|
|
formatDragObj(&ds->start, startBuf, VSIZE(startBuf) ),
|
|
formatDragObj(&ds->cur, curBuf, VSIZE(curBuf) ) );
|
|
}
|
|
#endif
|
|
|
|
#ifdef CPLUS
|
|
}
|
|
#endif
|